C#Reference PT BR
C#Reference PT BR
C#Reference PT BR
Documentação do C#
Introdução
Visão geral
Introdução à linguagem C# e ao .NET Framework
Tutoriais
Visão geral
Introdução à programação com C#
Escolha sua primeira lição
Hello world
Números em C#
Loops e branches
Coleções de lista
Trabalhar em seu ambiente local
Configure o seu ambiente
Números em C#
Loops e branches
Coleções de lista
Introdução às classes
Explorar o C# 6
Explorar a interpolação de cadeia de caracteres – interativo
Explorar a interpolação de cadeia de caracteres – em seu ambiente
Cenários avançados de interpolação de cadeia de caracteres
Atualizar interfaces de forma segura com membros de interface padrão
Criar funcionalidade mixin com métodos de interface padrão
Explore os índices e os intervalos
Trabalhar com tipos de referência nula
Atualizar um aplicativo para tipos de referência nula
Gerar e consumir fluxos assíncronos
Estender as funcionalidades de dados usando a correspondência de padrões
Aplicativo do Console
Cliente REST
Herança em C# e .NET
Trabalhando com LINQ
Usando atributos
Tour do C#
Introdução
Estrutura do programa
Tipos e variáveis
Expressões
Instruções
Classes e objetos
Structs
Matrizes
Interfaces
Enums
Delegados
Atributos
Novidades no C#
C# 8.0
C# 7.3
C# 7.2
C# 7.1
C# 7.0
C# 6
Histórico de versão do C#
Relações entre linguagem e estrutura
Considerações sobre versão e atualização
Conceitos de C#
Sistemas do tipo C#
Tipos de referência anuláveis
Descrever APIs anuláveis usando atributos e restrições
Namespaces
Tipos Básicos
Classes
Structs
Tuplas
Desconstruindo tuplas e outros tipos
Interfaces
Métodos
Expressões lambda
Propriedades
Indexadores
Descartes
Genéricos
Iterators
Delegados e eventos
Introdução a Delegados
System.Delegate e a palavra-chave delegado
Delegados Fortemente Tipados
Padrões Comuns para Delegados
Introdução a eventos
Padrões de evento .NET padrão
Padrão de Evento .NET Atualizado
Distinção entre Delegados e Eventos
LINQ (Consulta Integrada à Linguagem)
Visão geral de LINQ
Noções básicas sobre expressões de consulta
LINQ em C#
Escrever consultas LINQ em C#
Consultar uma coleção de objetos
Retornar uma consulta de um método
Armazenar os resultados de uma consulta na memória
Agrupar resultados de consultas
Criar um grupo aninhado
Executar uma subconsulta em uma operação de agrupamento
Agrupar resultados por chaves contíguas
Especificar filtros predicados dinamicamente em runtime
Executar junções internas
Executar junções agrupadas
Executar junções externas esquerdas
Ordenar os resultados de uma cláusula join
Unir usando chaves compostas
Executar operações de junção personalizadas
Manipular valores nulos em expressões de consulta
Tratar exceções em expressões de consulta
Programação assíncrona
Correspondência padrão
Escrever código eficiente e seguro
Árvores de expressão
Introdução à Árvores de expressão
Árvores de Expressão Explicadas
Tipos de Framework com suporte a árvores de expressão
Executando Expressões
Interpretando Expressões
Compilando Expressões
Traduzindo Expressões
Resumo
Interoperabilidade nativa
Documentando seu código
Controle de versão
Tópicos de instruções de C#
Índice de tópicos
Analisar cadeias de caracteres usando `String.Split`
Concatenar cadeias de caracteres
Converter uma cadeia de caracteres em um DateTime
Pesquisar cadeias de caracteres
Modificar o conteúdo de uma cadeia de caracteres
Comparar cadeias de caracteres
Converter usando a correspondência de padrões e os operadores is e as com
segurança
O SDK do .NET Compiler Platform (APIs do Roslyn)
Visão geral do SDK do .NET Compiler Platform (APIs do Roslyn)
Entender o modelo de API do compilador
Trabalhar com sintaxe
Trabalhar com semântica
Trabalhar com um workspace
Explorar código com o visualizador de sintaxe
Inícios Rápidos
Análise sintática
Análise semântica
Transformação de sintaxe
Tutoriais
Crie seu primeiro analisador e correção de código
Guia de Programação em C#
Visão geral
Por dentro de um programa em C#
O que há dentro de um Programa em C#
Hello World -- seu primeiro programa
Estrutura geral de um programa em C#
Nomes de identificadores
Convenções de codificação em C#
Main() e argumentos de linha de comando
Visão geral
Argumentos de linha de comando
Como: exibir argumentos de linha de comando
Valores de retorno de Main()
Conceitos de programação
Visão geral
Programação assíncrona com async e await
Visão geral
Modelo de programação assíncrona de tarefa
Passo a passo: acessar a Web usando async e await
Como estender o passo a passo assíncrono usando Task.WhenAll
Como fazer várias solicitações da Web em paralelo usando async e await
Tipos de retorno assíncronos
Fluxo de controle em programas assíncronos
Cancelamento de tarefas e processamento de tarefas concluídas
Visão geral
Cancelar uma tarefa assíncrona ou uma lista de tarefas
Cancelar tarefas assíncronas após um período
Cancelar tarefas assíncronas restantes após a conclusão de uma delas
Iniciar várias tarefas assíncronas e processá-las na conclusão
Tratar a reentrada em aplicativos assíncronos
Usando o Async para acessar arquivos
Atributos
Visão geral
Criando atributos personalizados
AttributeUsage
Acessando atributos usando reflexão
Como criar uma união do C/C++ usando atributos
Atributos comuns
Informações de chamador
Coleções
Covariância e contravariância
Visão geral
Variação em interfaces genéricas
Criar interfaces genéricas variáveis
Usar variação em interfaces para coleções genéricas
Variação em delegados
Usar variação em delegados
Usando variação para delegações genéricas Func e Action
Árvores de expressão
Visão geral
Como executar árvores de expressão
Como modificar árvores de expressão
Como usar árvores de expressão para compilar consultas dinâmicas
Depurar árvores de expressão no Visual Studio
Sintaxe do DebugView
Iterators
LINQ (Consulta Integrada à Linguagem)
Visão geral
Introdução a LINQ em C#
Introdução a consultas LINQ
LINQ e tipos genéricos
Operações de consulta LINQ básicas
Transformações de dados com LINQ
Relacionamentos de tipo em operações de consulta LINQ
Sintaxe de consulta e sintaxe de método em LINQ
funcionalidades do C# que dão suporte a LINQ
Passo a passo: Escrevendo consultas em C# (LINQ)
Visão geral de operadores de consulta padrão
Visão geral
Sintaxe de expressão da consulta para operadores de consulta padrão
Classificação de operadores de consulta padrão por maneira de execução
Classificar dados
Operações de conjunto
Filtrar dados
Operações de quantificador
Operações de projeção
Particionar dados
Operações de junção
Agrupar dados
Operações de geração
Operações de igualdade
Operações de elemento
Converter tipos de dados
Operações de concatenação
Operações de agregação
Objetos LINQ to
Visão geral
LINQ e cadeias de caracteres
Artigos de instruções
Como contar ocorrências de uma palavra em uma cadeia de caracteres (LINQ)
Como: consultar sentenças que contêm um conjunto específico de palavras
(LINQ)
Como: consultar caracteres em uma cadeia de caracteres (LINQ)
Como combinar consultas LINQ com expressões regulares
Como: localizar a diferença de conjunto entre duas listas (LINQ)
Como: classificar ou filtrar dados de texto por qualquer palavra ou campo
(LINQ)
Como: reordenar os campos de um arquivo delimitado (LINQ)
Como combinar e comparar coleções de cadeias de caracteres (LINQ)
Como: popular coleções de objetos de várias fontes (LINQ)
Como: dividir um arquivo em vários arquivos usando grupos (LINQ)
Como: unir o conteúdo de arquivos diferentes (LINQ)
Como computar valores de coluna em um arquivo de texto CSV (LINQ)
LINQ e reflexão
Como: consultar metadados de um assembly com reflexão (LINQ)
LINQ e diretórios de arquivos
Visão geral
Como: consultar arquivos com um atributo ou um nome especificado
Como: agrupar arquivos por extensão (LINQ)
Como: consultar o número total de bytes em um conjunto de pastas (LINQ)
Como comparar o conteúdo de duas pastas (LINQ)
Como: consultar o maior arquivo ou arquivos em uma árvore de diretório
(LINQ)
Como: consultar arquivos duplicados em uma árvore de diretório (LINQ)
Como: consultar o conteúdo de arquivos em uma pasta (LINQ)
Como: consultar um ArrayList com LINQ
Como adicionar métodos personalizados para consultas LINQ
LINQ to XML
Guia de introdução (LINQ to XML)
Visão geral de LINQ to XML
LINQ to XML e DOM
LINQ to XML e outras tecnologias XML
Guia de programação (LINQ to XML)
Visão geral da programação LINQ to XML
Árvores XML
Trabalhando com namespaces XML
Serializando árvores XML
Eixos LINQ to XML
Consultando árvores XML
Modificar árvores XML (LINQ to XML)
Desempenho (LINQ to XML)
Programação LINQ to XML avançada
Segurança LINQ to XML
Documentos XML de exemplo (LINQ to XML)
Referência (LINQ to XML)
LINQ to ADO.NET (página do portal)
Habilitando uma fonte de dados para consulta LINQ
Suporte do Visual Studio IDE e ferramentas para LINQ
Programação orientada a objeto
Reflexão
Serialização (C#)
Visão geral
Como: gravar dados de objeto em um arquivo XML
Como: ler dados de objeto de um arquivo XML
Passo a passo: persistir um objeto no Visual Studio
Instruções, expressões e operadores
Visão geral
Instruções
Expressões
Membros aptos para expressão
Funções anônimas
Visão geral
Expressões lambda
Como usar expressões lambda em uma consulta
Igualdade e comparações de igualdade
Comparações de igualdade
Como definir a igualdade de valor para um tipo
Como testar a igualdade de referência (identidade)
Tipos
Usar e definir tipos
Conversões cast e conversões de tipo
Conversão boxing e unboxing
Como: converter uma matriz de bytes em um int
Como: converter uma cadeia de caracteres em um número
Como: converter entre cadeias de caracteres hexadecimais e tipos numéricos
Usando o tipo dynamic
Passo a passo: criar e usar objetos dinâmicos (C# e Visual Basic)
Classes e structs
Visão geral
Classes
Objetos
Structs
Visão geral
Usando structs
Herança
Polimorfismo
Visão geral
Controle de versão com as palavras-chave override e new
Quando usar as palavras-chave override e new
Como substituir o método ToString
Membros
Visão geral dos membros
Classes e membros de classes abstract e sealed
Classes static e membros de classes static
Modificadores de acesso
Campos
Constantes
Como definir propriedades abstract
Como definir constantes em C#
Propriedades
Visão geral das propriedades
Usando propriedades
Propriedades de interface
Restringindo a acessibilidade ao acessador
Como declarar e usar propriedades de leitura/gravação
Propriedades autoimplementadas
Como implementar uma classe leve com propriedades autoimplementadas
Métodos
Visão geral dos métodos
Funções locais
Ref returns e ref locals
Parâmetros
Passando parâmetros
Passando parâmetros de tipo de valor
Passando parâmetros de tipo de referência
Como saber a diferença entre passar um struct e passar uma referência de classe
para um método
Variáveis Locais Tipadas Implicitamente
Como usar matrizes e variáveis locais de tipo implícito em uma expressão de
consulta
Métodos de extensão
Como implementar e chamar um método de extensão personalizado
Como criar um novo método para uma enumeração
Argumentos nomeados e opcionais
Como usar argumentos nomeados e opcionais na programação do Office
Construtores
Visão geral dos construtores
Usando construtores
Construtores de instância
Construtores particulares
Construtores estáticos
Como escrever um construtor de cópia
Finalizadores
Inicializadores de objeto e coleção
Como inicializar objetos usando um inicializador de objeto
Como: inicializar um dicionário com um inicializador de coleção
Tipos aninhados
Classes parciais e métodos
Tipos anônimos
Como retornar subconjuntos de propriedades de elementos em uma consulta
Interfaces
Visão geral
Implementação de interface explícita
Como: implementar membros de interface de forma explícita
Como: implementar membros de duas interfaces de forma explícita
Tipos de enumeração
Delegados
Visão geral
Usando delegados
Delegados com Métodos Nomeados vs. Métodos anônimos
Como: combinar delegados (delegados multicast) (Guia de Programação em C#)
Como: declarar, instanciar e usar um delegado
Matrizes
Visão geral
Matrizes como Objetos
Matrizes unidimensionais
Matrizes multidimensionais
Matrizes denteadas
Usar foreach com matrizes
Passar matrizes como argumentos
Matrizes de tipo implícito
Cadeias de caracteres
Programando com cadeias de caracteres
Como: determinar se uma cadeia de caracteres representa um valor numérico
Indexadores
Visão geral
Usando indexadores
Indexadores em interfaces
Comparação entre propriedades e indexadores
Eventos
Visão geral
Como: assinar eventos e cancelar a assinatura
Como: publicar eventos em conformidade com as diretrizes do .NET Framework
Como: acionar eventos de classe base em classes derivadas
Como: implementar eventos de interface
Como: implementar acessadores de eventos personalizados
Genéricos
Visão geral
Parâmetros de tipo genérico
Restrições a parâmetros de tipo
Classes genéricas
Interfaces genéricas
Métodos genéricos
Genéricos e matrizes
Delegados genéricos
Diferenças entre modelos C++ e genéricos C#
Genéricos em tempo de execução
Genéricos e reflexão
Genéricos e atributos
Namespaces
Visão geral
Usar namespaces
Como usar o My Namespace
Código não seguro e ponteiros
Visão geral e restrições
Buffers de tamanho fixo
Tipos de ponteiro
Visão geral
Conversões de ponteiro
Como usar ponteiros para copiar uma matriz de bytes
Comentários da documentação XML
Visão geral
Marcas recomendadas para comentários de documentação
Processando o arquivo XML
Delimitadores para marcações de documentação
Como: usar as funcionalidades de documentação XML
Referência de marcação de documentação
<c>
<code>
Atributo cref
<example>
<exception>
<include>
<list>
<para>
<param>
<paramref>
<permission>
<remarks>
<returns>
<see>
<seealso>
<summary>
<typeparam>
<typeparamref>
<value>
Exceções e manipulação de exceções
Visão geral
Usando exceções
Tratamento de Exceção
Criar e lançar exceções
Exceções geradas pelo compilador
Como: manipular uma exceção usando try/catch
Como: executar código de limpeza usando finally
Como: Capturar uma exceção não CLS
Sistema de arquivos e o Registro
Visão geral
Como: iterar em uma árvore de diretórios
Como: obter informações sobre arquivos, pastas e unidades
Como: criar um arquivo ou uma pasta
Como: copiar, excluir e mover arquivos e pastas
Como: fornecer uma caixa de diálogo de progresso para operações de arquivo
Como: gravar em um arquivo de texto
Como: ler de um arquivo de texto
Como: Ler um arquivo de texto uma linha de cada vez (Visual C#)
Como: Criar uma chave no Registro (Visual C#)
Interoperabilidade
Interoperabilidade .NET
Interoperabilidade .NET
Visão geral sobre interoperabilidade
Como: acessar objetos de interoperabilidade do Office usando recursos do Visual
C#
Como: usar propriedades indexadas na programação para interoperabilidade COM
Como: usar a invocação de plataforma para reproduzir um arquivo wave
Passo a passo: Programação do Office (C# e Visual Basic)
Exemplo de classe COM
Referência de linguagem
Visão geral
Configurar versão da linguagem
Palavras-chave de C#
Visão geral
Tipos internos
Tipos de valor
Recursos dos tipos de valor
Tipos numéricos integrais
Tipos numéricos de ponto flutuante
Conversões numéricas internas
bool
char
enum
struct
Tipos de valor anuláveis
Tipos de referência
Recursos dos tipos de referência
Tipos de referência internos
classe
interface
void
var
Tipos não gerenciados
Tabelas de referência de tipos
Tabela de tipos internos
Tabela de tipos de valor
Tabela de valores padrão
Tabela de formatação de resultados numéricos
Modificadores
Modificadores de acesso
Referência rápida
Níveis de acessibilidade
Domínio de acessibilidade
Restrições ao uso de níveis de acessibilidade
interno
particulares
protegidos
públicos
internos protegidos
privado protegido
abstract
async
const
evento
extern
em (modificador genérico)
novo (modificador de membro)
saída (modificador genérico)
override
readonly
sealed
static
unsafe
virtual
volatile
Palavras-chave de instrução
Categorias de instrução
Instruções de seleção
if-else
switch
Instruções de iteração
do
for
foreach, in
while
Instruções de atalho
break
continue
goto
return
Instruções para manipulação de exceções
throw
try-catch
try-finally
try-catch-finally
Contexto verificado e não verificado
Visão geral
checked
unchecked
Instrução fixed
Instrução lock
Parâmetros de método
Passando parâmetros
params
in (modificador de parâmetro)
ref
out (modificador de parâmetro)
Palavras-chave de namespace
namespace
using
Contextos de uso
Diretiva de uso
Diretiva using static
Instrução using
extern alias
Palavras-chave de teste de tipo
is
Tipo genérico de palavras-chave de restrição
nova restrição
onde
Palavras-chave de acesso
base
this
Palavras-chave literais
nulo
true
false
default
Palavras-chave contextuais
Referência rápida
adicionar
get
partial (tipo)
partial (método)
remove
set
when (condição de filtro)
Valor
yield
Palavras-chave de consulta
Referência rápida
Cláusula from
Cláusula where
Cláusula select
Cláusula group
into
Cláusula orderby
Cláusula join
Cláusula let
ascending
descending
em
é igual a
by
em
Operadores C#
Visão geral
Operadores aritméticos
Operadores lógicos boolianos
Operadores shift e bit a bit
Operadores de igualdade
Operadores de comparação
Operadores de acesso a membro
Operadores cast e teste de tipo
Operadores de conversões definidas pelo usuário
Operadores relacionados a ponteiro
Operadores de atribuição
+ e operadores +=
- e operadores -=
Operador ?:
! Operador do (null-forgiving)
?? e operadores ??=
Operador =>
Operador ::
operador await
operador padrão
operador de representante
operador nameof
Operador new
Operador sizeof
operador stackalloc
operadores true e false
Sobrecarga de operador
Caracteres especiais de C#
Visão geral
Interpolação de cadeia de caracteres -- $
@ – identificador textual
Diretivas de pré-processador do C#
Visão geral
#if
#else
#elif
#endif
#define
#undef
#warning
#error
#line
#region
#endregion
#pragma
#pragma warning
#pragma checksum
Opções do compilador de C#
Visão geral
Compilando pela linha de comando com csc.exe
Como configurar variáveis de ambiente para a linha de comando do Visual Studio
Opções do compilador de C# listadas por categoria
Opções do compilador de C# listadas em ordem alfabética
@
-addmodule
-appconfig
-baseaddress
-bugreport
-checked
-codepage
-debug
-define
-delaysign
-deterministic
-doc
-errorreport
-filealign
-fullpaths
/help, /?
-highentropyva
-keycontainer
-keyfile
-langversion
-lib
-link
-linkresource
-main
-moduleassemblyname
-noconfig
-nologo
-nostdlib
-nowarn
-nowin32manifest
-optimize
-out
-pathmap
-pdb
-platform
-preferreduilang
-publicsign
-recurse
-reference
-refout
-refonly
-resource
-subsystemversion
-target
-target:appcontainerexe
-target:exe
-target:library
-target:module
-target:winexe
-target:winmdobj
-unsafe
-utf8output
-warn
-warnaserror
-win32icon
-win32manifest
-win32res
Erros do compilador de C#
Especificação de rascunho do C# 6.0
Propostas do C# 7.0 - 8.0
Passo a passo
Introdução ao C#
31/10/2019 • 3 minutes to read • Edit Online
Esta seção fornece breves tutoriais simples que permitem que você compile rapidamente um aplicativo usando C#
e .NET Core. Há tópicos com introdução ao Visual Studio 2017 e ao Visual Studio Code. Esses artigos supõem
experiência prévia em programação. Se você for iniciante em programação, experimente nossos tutoriais
interativos de Introdução ao C#.
Os seguintes tópicos estão disponíveis:
Introdução à linguagem C# e ao .NET Framework
Fornece uma visão geral da linguagem C# e da plataforma .NET.
Compilando um aplicativo Olá, Mundo em C# com o .NET Core no Visual Studio 2017
O Visual Studio permite codificar, compilar, executar, depurar, criar um perfil e publicar seus aplicativos de
um ambiente de desenvolvimento integrado para Windows ou Mac.
O tópico permite criar e executar um aplicativo Olá, Mundo simples e, em seguida, modificá-lo para
executar um aplicativo Olá, Mundo ligeiramente mais interativo. Quando terminar de compilar e executar
seu aplicativo, você também poderá aprender como depurá-lo e como publicá-lo para que ele possa ser
executado em qualquer plataforma com suporte do .NET Core.
Compilando uma biblioteca de classes com C# e .NET Core no Visual Studio 2017
Uma biblioteca de classes permite definir tipos e membros de tipo que podem ser chamados de outro
aplicativo. Este tópico permite criar uma biblioteca de classes com um único método que determina se uma
cadeia de caracteres começa com um caractere maiúsculo. Após criar a biblioteca, você pode desenvolver
um teste de unidade para garantir que ela funciona conforme o esperado e, em seguida, você pode torná-la
disponível para aplicativos que desejam consumi-la.
Introdução ao Código do Visual Studio e C#
O Visual Studio Code é um editor de código gratuito e otimizado para compilação e depuração de
aplicativos Web e de nuvem modernos. Ele dá suporte ao IntelliSense e está disponível para Windows,
Linux e macOS.
Este tópico mostra como criar e executar um aplicativo Olá, Mundo simples com o código do Visual Studio
e do .NET Core.
Seções relacionadas
Guia de Programação em C#
Fornece informações sobre conceitos de programação C# e descreve como executar várias tarefas no C#.
Referência de C#
Fornece informações de referência detalhadas sobre palavras-chave, operadores, diretivas de pré-
processamento, opções do compilador, erros do compilador e avisos do #C.
Explicações Passo a Passo
Fornece links para explicações passo a passo sobre programação em C# e uma breve descrição sobre cada
explicação passo a passo.
Consulte também
Desenvolvimento em C# com o Visual Studio
Introdução à linguagem C# e ao .NET Framework
25/11/2019 • 10 minutes to read • Edit Online
C# é uma linguagem elegante, orientada a objeto e fortemente tipada, que permite que os desenvolvedores criem
uma variedade de aplicativos robustos e seguros executados no .NET Framework. Você pode usar C# para criar
aplicativos de cliente do Windows, serviços Web XML, componentes distribuídos, aplicativos cliente-servidor,
aplicativos de banco de dados e muito, muito mais. O Visual C# fornece um editor de código avançado, designers
de interface do usuário convenientes, depurador integrado e muitas outras ferramentas para facilitar o
desenvolvimento de aplicativos com base na linguagem C# e no .NET Framework.
NOTE
A documentação do Visual C# considera que você já tenha conhecimento dos conceitos básicos de programação. Se você é
principiante, convém explorar o Visual C# Express, que está disponível na Web. Você também pode tirar proveito de livros e
recursos da Web sobre C# para aprender técnicas de programação práticas.
Linguagem C#
A sintaxe de C# é altamente expressiva, mas também é simples e fácil de aprender. A sintaxe de chaves de C# será
instantaneamente reconhecível para qualquer pessoa familiarizada com C, C++ ou Java. Normalmente, os
desenvolvedores que conhecem qualquer uma dessas linguagens são capazes de começar a trabalhar de forma
produtiva em C# dentro de um período muito curto. A sintaxe de C# simplifica muitas complexidades de C++ e
fornece recursos poderosos como tipos de valor anulável, enumerações, delegados, expressões lambda e acesso
direto à memória, que não existem em Java. C# oferece suporte a tipos e métodos genéricos, o que proporciona
mais segurança e desempenho para os tipos, e iteradores, que permitem aos implementadores das classes de
coleção definir os comportamentos personalizados da iteração simples de usar pelo código do cliente. As
expressões LINQ (Consulta Integrada à Linguagem) tornam a consulta fortemente tipada uma construção de
linguagem de primeira classe.
Por ser uma linguagem orientada a objeto, o C# oferece suporte aos conceitos de encapsulamento, herança e
polimorfismo. Todas as variáveis e métodos, incluindo o método Main , o ponto de entrada do aplicativo, são
encapsulados em definições de classe. Uma classe pode herdar diretamente de uma classe pai, mas pode
implementar qualquer quantidade de interfaces. Métodos que substituem métodos virtuais em uma classe pai
exigem a palavra-chave override como uma forma de evitar uma redefinição acidental. Em C#, um struct é como
uma classe simplificada; é um tipo alocado na pilha que pode implementar interfaces, mas não oferece suporte a
herança.
Além desses princípios básicos orientados a objeto, C# facilita o desenvolvimento de componentes de software
por meio de várias construções de linguagem inovadoras, incluindo o seguinte:
Assinaturas de método encapsulado chamadas de delegados, que permitem as notificações de eventos
fortemente tipados.
Propriedades, que servem como acessadores para variáveis de membro privado.
Atributos, que fornecem metadados declarativos sobre os tipos no tempo de execução.
Comentários embutidos da documentação XML.
LINQ (Consulta Integrada à Linguagem) que fornece recursos de consulta internos em várias fontes de
dados.
Se você precisar interagir com outros softwares do Windows, como objetos COM ou DLLs Win32 nativas, faça
isso em C# através de um processo denominado "Interoperabilidade". A interoperabilidade permite que
programas em C# façam quase tudo que um aplicativo C++ nativo pode fazer. C# oferece suporte até mesmo para
ponteiros, e o conceito de código "não seguro" para os casos nos quais o acesso direto à memória é absolutamente
essencial.
O processo de compilação de C# é simples comparado ao C e C++, e mais flexível do que em Java. Não há
arquivos de cabeçalho separado, e nenhum requisito de que os métodos e os tipos sejam declarados em uma
ordem específica. Um arquivo de código-fonte de C# pode definir qualquer quantidade de classes, estruturas,
interfaces e eventos.
Veja a seguir recursos adicionais de C#:
Para uma boa introdução geral à linguagem, consulte o Capítulo 1 da Especificação da linguagem C#.
Para obter informações detalhadas sobre aspectos específicos da linguagem C#, consulte a Referência de
C#.
Para saber mais sobre LINQ, confira LINQ (Consulta Integrada à Linguagem).
Consulte também
Introdução com VisualC#
Tutoriais do C#
04/11/2019 • 7 minutes to read • Edit Online
Bem-vindo aos tutoriais do C#. Eles começam com lições interativas que podem ser executadas em seu navegador.
Os tutoriais posteriores e os tutoriais mais avançados ajudam você a trabalhar com as ferramentas de
desenvolvimento do .NET para criar programas em C# em seu computador.
Olá, Mundo
No tutorial Olá, Mundo, você criará o programa C# mais básico. Você explorará o tipo string e como trabalhar
com texto.
Números em C#
No tutorial Números em C#, você aprenderá como os computadores armazenam números e como executar
cálculos com diferentes tipos de número. Você aprenderá os conceitos básicos de arredondamento e como
executar cálculos matemáticos usando C#. Este tutorial também está disponível para execução local no seu
computador.
Esse tutorial pressupõe a conclusão da lição Olá, Mundo.
Ramificações e loops
O tutorial Branches e loops ensina os conceitos básicos da seleção de diferentes caminhos de execução de código
com base nos valores armazenados em variáveis. Você aprenderá os conceitos básicos do fluxo de controle, que
são os fundamentos de como os programas tomam decisões e escolhem ações diferentes. Este tutorial também
está disponível para execução local no seu computador.
Esse tutorial pressupõe a conclusão das lições Olá, Mundo e Números em C#.
Coleções de lista
A lição Coleções de lista fornece um tour pelo tipo Coleções de lista que armazena as sequências de dados. Você
aprenderá a adicionar e remover itens, pesquisar itens e classificar listas. Você explorará os diferentes tipos de
listas. Este tutorial também está disponível para execução local no seu computador.
Esse tutorial pressupõe a conclusão das lições listadas acima.
Tutoriais gerais
Os tutoriais a seguir permitem que você compile programas em C# usando .NET Core:
Aplicativo de console: demonstra a E/S de console, a estrutura de um aplicativo de console e os conceitos
básicos do modelo de programação assíncrono baseado em tarefa.
Cliente REST: demonstra as comunicações da Web, serialização JSON e recursos orientados a objeto na
linguagem C#.
Herança em C# e .NET: demonstra herança em C#, incluindo o uso de herança para definir classes derivadas,
classes base e classes base abstratas.
Trabalhar com LINQ: demonstra muitos recursos do LINQ e os elementos de linguagem que dão suporte a ele.
Usando atributos: demonstra como criar e usar atributos em C#.
O tutorial Interpolação de cadeia de caracteres mostra como inserir valores em uma cadeia de caracteres. Você
aprenderá como criar uma cadeia de caracteres interpolada com expressões de C# incorporadas e como
controlar a aparência do texto dos resultados de expressão na cadeia de caracteres de resultado. Este tutorial
também está disponível para execução local no seu computador.
Introdução ao C#
04/11/2019 • 4 minutes to read • Edit Online
Bem-vindo aos tutoriais de introdução ao C#. Eles começam com lições interativas que podem ser executadas em
seu navegador. Você pode aprender as noções básicas C# da C# série de vídeos 101 antes de iniciar essas lições
interativas.
As primeiras lições explicam os conceitos de C# usando pequenos snippets de código. Você aprenderá os
conceitos básicos da sintaxe de C# e como trabalhar com tipos de dados como cadeias de caracteres, números e
valores boolianos. É tudo interativo e você começará a gravar e executar o código em questão de minutos. Estas
primeiras lições não exigem conhecimento prévio de programação ou da linguagem C#.
Todos os tutoriais de introdução posteriores à lição Olá, Mundo estão disponíveis por meio da experiência de
navegador online ou em seu próprio ambiente de desenvolvimento local. No final de cada tutorial, você decidirá
se deseja continuar com a próxima lição online ou no próprio computador. Há links para ajudar você a configurar
seu ambiente e continuar com o próximo tutorial no computador.
Olá, Mundo
No tutorial Olá, Mundo, você criará o programa C# mais básico. Você explorará o tipo string e como trabalhar
com texto.
Números em C#
No tutorial Números em C#, você aprenderá como os computadores armazenam números e como executar
cálculos com diferentes tipos de número. Você aprenderá os conceitos básicos de arredondamento e como
executar cálculos matemáticos usando C#. Este tutorial também está disponível para execução local no seu
computador.
Esse tutorial pressupõe a conclusão da lição Olá, Mundo.
Ramificações e loops
O tutorial Branches e loops ensina os conceitos básicos da seleção de diferentes caminhos de execução de código
com base nos valores armazenados em variáveis. Você aprenderá os conceitos básicos do fluxo de controle, que
são os fundamentos de como os programas tomam decisões e escolhem ações diferentes. Este tutorial também
está disponível para execução local no seu computador.
Esse tutorial pressupõe a conclusão das lições Olá, Mundo e Números em C#.
Coleções de lista
A lição Coleções de lista fornece um tour pelo tipo Coleções de lista que armazena as sequências de dados. Você
aprenderá a adicionar e remover itens, pesquisar itens e classificar listas. Você explorará os diferentes tipos de
listas. Este tutorial também está disponível para execução local no seu computador.
Esse tutorial pressupõe a conclusão das lições listadas acima.
Introdução às classes
Este tutorial final está disponível apenas para execução no seu computador usando seu próprio ambiente de
desenvolvimento local e o .NET Core. Você compilará um aplicativo de console e verá os recursos básicos
orientados para objeto que fazem parte da linguagem C#.
Este tutorial pressupõe a conclusão dos tutoriais de introdução online e a instalação do SDK do .NET Core e do
Visual Studio Code.
Familiarize-se com as ferramentas de
desenvolvimento .NET
23/10/2019 • 3 minutes to read • Edit Online
A primeira etapa para executar um tutorial em seu computador é configurar um ambiente de desenvolvimento.
O tutorial do .NET Olá, mundo em 10 minutos tem instruções para configurar seu ambiente de desenvolvimento
local no Windows, Linux ou MacOS.
Como alternativa, você pode instalar o SDK do .NET Core e o Visual Studio Code.
Números em C#
No tutorial Números em C#, você aprenderá como os computadores armazenam números e como executar
cálculos com diferentes tipos de número. Você aprenderá os conceitos básicos de arredondamento e como
executar cálculos matemáticos usando C#.
Esse tutorial pressupõe a conclusão da lição Olá, Mundo.
Ramificações e loops
O tutorial Branches e loops ensina os conceitos básicos da seleção de diferentes caminhos de execução de código
com base nos valores armazenados em variáveis. Você aprenderá os conceitos básicos do fluxo de controle, que
são os fundamentos de como os programas tomam decisões e escolhem ações diferentes.
Esse tutorial pressupõe a conclusão das lições Olá, Mundo e Números em C#.
Coleções de lista
A lição Coleções de lista fornece um tour pelo tipo Coleções de lista que armazena as sequências de dados. Você
aprenderá a adicionar e remover itens, pesquisar itens e classificar listas. Você explorará os diferentes tipos de
listas.
Esse tutorial pressupõe a conclusão das lições listadas acima.
Introdução às classes
Esse tutorial final de introdução ao C# está disponível apenas para execução no seu computador usando seu
próprio ambiente de desenvolvimento local e o .NET Core. Você compilará um aplicativo de console e verá os
recursos básicos orientados para objeto que fazem parte da linguagem C#.
Manipular números de ponto flutuante e integrais
em C#
08/11/2019 • 15 minutes to read • Edit Online
Este tutorial ensina sobre os tipos numéricos em C# de maneira interativa. Você escreverá pequenas
quantidades de código, depois compilará e executará esse código. O tutorial contém uma série de lições que
exploram números e operações matemáticas em C#. Estas lições ensinam os princípios básicos da linguagem
C#.
Este tutorial espera que você tenha um computador que possa usar para desenvolvimento. O tutorial do .NET
Olá, mundo em 10 minutos tem instruções para configurar seu ambiente de desenvolvimento local no Windows,
Linux ou MacOS. Uma visão geral dos comandos que você usará está em Familiarize-se com as ferramentas de
desenvolvimento, com links para obter mais detalhes.
Abra Program.cs em seu editor favorito e substitua a linha Console.WriteLine("Hello World!"); pelo seguinte:
int a = 18;
int b = 6;
int c = a + b;
Console.WriteLine(c);
// multiplication
c = a * b;
Console.WriteLine(c);
// division
c = a / b;
Console.WriteLine(c);
TIP
À medida que explora C# (ou qualquer linguagem de programação), você cometerá erros ao escrever o código. O
compilador encontrará esses erros e os reportará a você. Quando a saída contiver mensagens de erro, analise
atentamente o código de exemplo e o código em sua janela para ver o que deve ser corrigido. Esse exercício ajudará você a
conhecer a estrutura do código C#.
Você terminou a primeira etapa. Antes de iniciar a próxima seção, vamos passar o código atual para um método
separado. Isso facilita o começo do trabalho com um exemplo novo. Renomeie seu método Main como
WorkingWithIntegers e escreva um novo método Main que chama WorkingWithIntegers . Quando você terminar,
seu código deverá ter a seguinte aparência:
using System;
namespace NumbersInCSharp
{
class Program
{
static void WorkingWithIntegers()
{
int a = 18;
int b = 6;
// addition
int c = a + b;
Console.WriteLine(c);
// subtraction
c = a - b;
Console.WriteLine(c);
// multiplication
c = a * b;
Console.WriteLine(c);
// division
c = a / b;
Console.WriteLine(c);
}
//WorkingWithIntegers();
O // inicia um comentário em C#. Os comentários são qualquer texto que você queira manter em seu código-
fonte, mas não queria executar como código. O compilador não gera qualquer código executável a partir dos
comentários.
A linguagem C# define a precedência de operações matemáticas diferentes com regras consistentes às regras
que você aprendeu em matemática. Multiplicação e divisão têm precedência sobre adição e subtração. Explore
isso adicionando o seguinte código ao seu método Main e executando dotnet run :
int a = 5;
int b = 4;
int c = 2;
int d = a + b * c;
Console.WriteLine(d);
Explore mais, combinando várias operações diferentes. Adicione algo parecido com as seguintes linhas na parte
inferior de seu método Main . Tente dotnet run novamente.
d = (a + b) - 6 * c + (12 * 4) / 3 + 12;
Console.WriteLine(d);
Talvez você tenha observado um comportamento interessante com relação aos números inteiros. A divisão de
inteiros sempre produz um resultado inteiro, mesmo quando você espera que o resultado inclua uma parte
decimal ou fracionária.
Se você ainda não viu esse comportamento, tente o seguinte código ao final de seu método Main :
int e = 7;
int f = 4;
int g = 3;
int h = (e + f) / g;
Console.WriteLine(h);
namespace NumbersInCSharp
{
class Program
{
static void WorkingWithIntegers()
{
int a = 18;
int b = 6;
// addition
int c = a + b;
Console.WriteLine(c);
// subtraction
c = a - b;
Console.WriteLine(c);
// multiplication
c = a * b;
Console.WriteLine(c);
// division
c = a / b;
Console.WriteLine(c);
}
d = (a + b) * c;
Console.WriteLine(d);
d = (a + b) - 6 * c + (12 * 4) / 3 + 12;
Console.WriteLine(d);
int e = 7;
int f = 4;
int g = 3;
int h = (e + f) / g;
Console.WriteLine(h);
}
OrderPrecedence();
}
}
}
O tipo de inteiro C# difere do inteiros matemáticos de outra forma: o tipo int tem limites mínimo e máximo.
Adicione este código ao seu método Main para ver esses limites:
Se um cálculo produzir um valor que excede esses limites, você terá uma condição de estouro negativo ou
estouro. A resposta parece quebrar de um limite para o outro. Adicione estas duas linhas ao seu método Main
para ver um exemplo:
Observe que a resposta é muito próxima do mínimo inteiro (negativo). É o mesmo que min + 2 . A operação de
adição estourou os valores permitidos para números inteiros. A resposta é um número negativo muito grande,
pois um estouro "envolve" do maior valor de inteiro possível para o menor.
Há outros tipos numéricos com limites e precisão diferentes que você usaria quando o tipo int não atendesse
às suas necessidades. Vamos explorá-los na sequência.
Novamente, vamos passar o código que você escreveu nesta seção para um método separado. Nomeie-o como
TestLimits .
double a = 5;
double b = 4;
double c = 2;
double d = (a + b) / c;
Console.WriteLine(d);
Observe que a resposta inclui a parte decimal do quociente. Experimente uma expressão ligeiramente mais
complicada com duplos:
double e = 19;
double f = 23;
double g = 8;
double h = (e + f) / g;
Console.WriteLine(h);
O intervalo de um valor duplo é muito maior do que valores inteiros. Experimente o código a seguir abaixo do
código que você escreveu até o momento:
Esses valores são impressos em notação científica. O número à esquerda do E é o significando. O número à
direita é o expoente, como uma potência de 10.
Assim como os números decimais em matemática, os duplos em C# podem ter erros de arredondamento.
Experimente esse código:
Você sabe que a repetição de 0.3 não é exatamente o mesmo que 1/3 .
Desafio
Experimente outros cálculos com números grandes, números pequenos, multiplicação e divisão usando o tipo
double . Experimente cálculos mais complicados.
Após algum tempo no desafio, pegue o código que você escreveu e coloque-o em um novo método. Chame
esse novo método de WorkWithDoubles .
Observe que o intervalo é menor do que o tipo double . Veja a precisão maior com o tipo decimal
experimentando o código a seguir:
double a = 1.0;
double b = 3.0;
Console.WriteLine(a / b);
decimal c = 1.0M;
decimal d = 3.0M;
Console.WriteLine(c / d);
O sufixo M nos números é o modo como você indica que uma constante deve usar o tipo decimal .
Observe que o cálculo usando o tipo decimal tem mais dígitos à direita da vírgula decimal.
Desafio
Agora que você viu os diferentes tipos numéricos, escreva um código que calcula a área de um círculo cujo raio é
de 2,50 centímetros. Lembre-se de que a área de um círculo é o quadrado do raio multiplicado por PI. Uma dica:
o .NET contém uma constante para PI, Math.PI, que você pode usar para esse valor.
Você deve obter uma resposta entre 19 e 20. Confira sua resposta analisando o código de exemplo finalizado no
GitHub
Experimente outras fórmulas, se quiser.
Você concluiu o início rápido "Números em C#". Continue com o início rápido Branches e loops em seu próprio
ambiente de desenvolvimento.
Saiba mais sobre os números em C# nos tópicos a seguir:
Tipos numéricos inteiros
Tipos numéricos de ponto flutuante
Conversões numéricas internas
Saiba mais sobre lógica condicional com instruções
branch e loop
08/11/2019 • 15 minutes to read • Edit Online
Este tutorial ensina a escrever código que examina variáveis e muda o caminho de execução com base nessas
variáveis. Escreva o código em C# e veja os resultados da compilação e da execução. O tutorial contém uma
série de lições que exploram construções de branches e loops em C#. Estas lições ensinam os princípios básicos
da linguagem C#.
Este tutorial espera que você tenha um computador que possa usar para desenvolvimento. O tutorial do .NET
Olá, mundo em 10 minutos tem instruções para configurar seu ambiente de desenvolvimento local no Windows,
Linux ou MacOS. Uma visão geral dos comandos que você usará está em Familiarize-se com as ferramentas de
desenvolvimento, com links para obter mais detalhes.
Esse comando cria um novo aplicativo de console .NET Core no diretório atual.
Abra Program.cs em seu editor favorito e substitua a linha Console.WriteLine("Hello World!"); pelo seguinte
código:
int a = 5;
int b = 6;
if (a + b > 10)
Console.WriteLine("The answer is greater than 10.");
Experimente este código digitando dotnet run na sua janela do console. Você deve ver a mensagem "A resposta
é maior do que 10." impressa no console.
Modifique a declaração de b para que a soma seja inferior a 10:
int b = 3;
Digite dotnet run novamente. Como a resposta é inferior a 10, nada é impresso. A condição que você está
testando é falsa. Não há qualquer código para execução, porque você escreveu apenas uma das ramificações
possíveis para uma instrução if : a ramificação verdadeira.
TIP
À medida que explora C# (ou qualquer linguagem de programação), você cometerá erros ao escrever o código. O
compilador encontrará e reportará esses erros. Verifique atentamente a saída do erro e o código que gerou o erro. O erro
do compilador geralmente pode ajudá-lo a localizar o problema.
Este primeiro exemplo mostra o poder dos tipos if e Booliano. Um Booliano é uma variável que pode ter um
dos dois valores: true ou false . C# define um tipo especial, bool para variáveis Boolianas. A instrução if
verifica o valor de um bool . Quando o valor é true , a instrução após if é executada. Caso contrário, é
ignorada.
Esse processo de verificação de condições e execução de instruções com base nessas condições é muito eficiente.
int a = 5;
int b = 3;
if (a + b > 10)
Console.WriteLine("The answer is greater than 10");
else
Console.WriteLine("The answer is not greater than 10");
A instrução após a palavra-chave else é executada somente quando a condição que estiver sendo testada for
false . A combinação de if e else com condições Boolianas fornece todos os recursos que você precisa para
lidar com uma condição true e false .
IMPORTANT
O recuo sob as instruções if e else é para leitores humanos. A linguagem C# não considera recuos ou espaços em
branco como significativos. A instrução após a palavra-chave if ou else será executada com base na condição. Todos
os exemplos neste tutorial seguem uma prática comum para recuar linhas com base no fluxo de controle de instruções.
Como o recuo não é significativo, você precisa usar { e } para indicar quando você quer que mais de uma
instrução faça parte do bloco executado condicionalmente. Os programadores em C# normalmente usam essas
chaves em todas as cláusulas if e else . O exemplo a seguir é igual ao que você acabou de criar. Modifique o
código acima para coincidir com o código a seguir:
int a = 5;
int b = 3;
if (a + b > 10)
{
Console.WriteLine("The answer is greater than 10");
}
else
{
Console.WriteLine("The answer is not greater than 10");
}
TIP
No restante deste tutorial, todos os exemplos de código incluem as chaves, seguindo as práticas aceitas.
Você pode testar condições mais complicadas. Adicione o seguinte código ao seu método Main após o código
que você escreveu até agora:
int c = 4;
if ((a + b + c > 10) && (a == b))
{
Console.WriteLine("The answer is greater than 10");
Console.WriteLine("And the first number is equal to the second");
}
else
{
Console.WriteLine("The answer is not greater than 10");
Console.WriteLine("Or the first number is not equal to the second");
}
O símbolo == testa a igualdade. Usar == distingue o teste de igualdade de atribuição, que você viu em a = 5 .
O && representa "e". Isso significa que as duas condições devem ser verdadeiras para executar a instrução no
branch verdadeiro. Estes exemplos também mostram que você pode ter várias instruções em cada branch
condicional, desde que você coloque-as entre { e } .
Você também pode usar || para representar "ou". Adicione o código a seguir após o que você escreveu até o
momento:
Modifique os valores de a , b e c e alterne entre && e || para explorar. Você obterá mais compreensão de
como os operadores && e || funcionam.
Você terminou a primeira etapa. Antes de iniciar a próxima seção, vamos passar o código atual para um método
separado. Isso facilita o começo do trabalho com um exemplo novo. Renomeie seu método Main como
ExploreIf e escreva um novo método Main que chama ExploreIf . Quando você terminar, seu código deverá
parecer com isto:
using System;
namespace BranchesAndLoops
{
class Program
{
static void ExploreIf()
{
int a = 5;
int b = 3;
if (a + b > 10)
{
Console.WriteLine("The answer is greater than 10");
}
else
{
Console.WriteLine("The answer is not greater than 10");
}
int c = 4;
if ((a + b + c > 10) && (a > b))
{
Console.WriteLine("The answer is greater than 10");
Console.WriteLine("And the first number is greater than the second");
}
else
{
Console.WriteLine("The answer is not greater than 10");
Console.WriteLine("Or the first number is not greater than the second");
}
Comente a chamada para ExploreIf() . Isso tornará a saída menos congestionada enquanto você trabalha nesta
seção:
//ExploreIf();
O // inicia um comentário em C#. Os comentários são qualquer texto que você queira manter em seu código-
fonte, mas não queria executar como código. O compilador não gera qualquer código executável a partir dos
comentários.
A instrução while verifica uma condição e executa a instrução, ou bloco de instruções, após o while . Ela
verifica repetidamente a condição e executa essas instruções até que a condição seja falsa.
Há outro operador novo neste exemplo. O ++ após a variável counter é o operador increment. Ele adiciona 1
ao valor de counter e armazena esse valor na variável counter .
IMPORTANT
Verifique se a condição de loop while muda para false ao executar o código. Caso contrário, crie um loop infinito, para
que seu programa nunca termine. Isso não é demonstrado neste exemplo, porque você tem que forçar o programa a
encerrar usando CTRL-C ou outros meios.
O loop while testa a condição antes de executar o código seguindo while . O loop do ... while executa o
código primeiro e, em seguida, verifica a condição. O loop do while é mostrado no código a seguir:
int counter = 0;
do
{
Console.WriteLine($"Hello World! The counter is {counter}");
counter++;
} while (counter < 10);
Ele faz o mesmo trabalho que o loop while e o loop do que você já usou. A instrução for tem três partes que
controlam o modo como ela funciona.
A primeira parte é o inicializador for: int index = 0; declara que index é a variável do loop, e define seu
valor inicial como 0 .
A parte central é a condição for: index < 10 declara que este loop for continuará sendo executado desde que
o valor do contador seja inferior a 10.
A parte final é o iterador for: index++ especifica como modificar a variável de loop depois de executar o bloco
após a instrução for . Aqui, ela especifica que index deve ser incrementado com 1 sempre que o bloco for
executado.
Experimente você mesmo. Tente o seguinte:
Altere o inicializador para iniciar em um valor diferente.
Altere a condição para parar em um valor diferente.
Quando terminar, vamos escrever um código para usar o que você aprendeu.
Tente você mesmo. Depois verifique como você fez. Você deve obter 63 como resposta. Veja uma resposta
possível exibindo o código completo no GitHub.
Você concluiu o tutorial "branches e loops".
Continue com o tutorial Matrizes e coleções em seu próprio ambiente de desenvolvimento.
Saiba mais sobre esses conceitos nestes tópicos:
Instrução If e else
Instrução while
Instrução Do
Instrução for
Saiba como gerenciar coleções de dados usando o
tipo de lista genérico
23/10/2019 • 9 minutes to read • Edit Online
Este tutorial de introdução fornece uma introdução à linguagem C# e os conceitos básicos da classe List<T>.
Este tutorial espera que você tenha um computador que possa usar para desenvolvimento. O tutorial do .NET
Olá, mundo em 10 minutos tem instruções para configurar seu ambiente de desenvolvimento local no Windows,
Linux ou MacOS. Uma visão geral de comandos que você usará está em Familiarizar-se com as ferramentas de
desenvolvimento, com links para obter mais detalhes.
using System;
using System.Collections.Generic;
namespace list_tutorial
{
class Program
{
static void Main(string[] args)
{
var names = new List<string> { "<name>", "Ana", "Felipe" };
foreach (var name in names)
{
Console.WriteLine($"Hello {name.ToUpper()}!");
}
}
}
}
Substitua <name> pelo seu nome. Salve Program.cs. Digite dotnet run na janela de console para testá-lo.
Você criou uma lista de cadeias de caracteres, adicionou três nomes a essa lista e imprimiu os nomes em
MAIÚSCULAS. Você está usando conceitos que aprendeu em tutoriais anteriores para executar um loop pela
lista.
O código para exibir nomes utiliza o recurso de interpolação de cadeia de caracteres. Quando você precede um
string com o caractere $ , pode inserir o código C# na declaração da cadeia de caracteres. A cadeia de
caracteres real substitui esse código C# pelo valor gerado. Neste exemplo, ela substitui o {name.ToUpper()} por
cada nome, convertido em letras maiúsculas, pois você chamou o método ToUpper.
Vamos continuar explorando.
Console.WriteLine();
names.Add("Maria");
names.Add("Bill");
names.Remove("Ana");
foreach (var name in names)
{
Console.WriteLine($"Hello {name.ToUpper()}!");
}
Você adicionou mais dois nomes ao final da lista. Também removeu um. Salve o arquivo e digite dotnet run
para testá-lo.
O List<T> também permite fazer referência a itens individuais por índice. Coloque o índice entre os tokens [ e
] após o nome da lista. C# usa 0 para o primeiro índice. Adicione este código diretamente abaixo do código
que você acabou de adicionar e teste-o:
Você não pode acessar um índice além do fim da lista. Lembre-se de que os índices começam com 0, portanto, o
maior índice válido é uma unidade a menos do que o número de itens na lista. Você pode verificar há quanto
tempo a lista está usando a propriedade Count. Adicione o código a seguir ao final do método Main:
}
Os itens em sua lista também podem ser classificados. O método Sort classifica todos os itens na lista na ordem
normal (em ordem alfabética, no caso de cadeias de caracteres). Adicione este código à parte inferior de nosso
método Main :
names.Sort();
foreach (var name in names)
{
Console.WriteLine($"Hello {name.ToUpper()}!");
}
Salve o arquivo e digite dotnet run para experimentar a versão mais recente.
Antes de iniciar a próxima seção, vamos passar o código atual para um método separado. Isso facilita o começo
do trabalho com um exemplo novo. Renomeie seu método Main como WorkingWithStrings e escreva um novo
método Main que chama WorkingWithStrings . Quando você terminar, seu código deverá parecer com isto:
using System;
using System.Collections.Generic;
namespace list_tutorial
{
class Program
{
static void Main(string[] args)
{
WorkingWithStrings();
}
Console.WriteLine();
names.Add("Maria");
names.Add("Bill");
names.Remove("Ana");
foreach (var name in names)
{
Console.WriteLine($"Hello {name.ToUpper()}!");
}
names.Sort();
foreach (var name in names)
{
Console.WriteLine($"Hello {name.ToUpper()}!");
}
}
}
}
Isso cria uma lista de números inteiros e define os primeiros dois inteiros como o valor 1. Estes são os dois
primeiros valores de uma sequência Fibonacci, uma sequência de números. Cada número Fibonacci seguinte é
encontrado considerando a soma dos dois números anteriores. Adicione este código:
var previous = fibonacciNumbers[fibonacciNumbers.Count - 1];
var previous2 = fibonacciNumbers[fibonacciNumbers.Count - 2];
fibonacciNumbers.Add(previous + previous2);
TIP
Para se concentrar apenas nesta seção, comente o código que chama WorkingWithStrings(); . Coloque apenas dois
caracteres / na frente da chamada, desta forma: // WorkingWithStrings(); .
Desafio
Veja se você consegue combinar alguns dos conceitos desta lição e de lições anteriores. Expanda o que você
compilou até o momento com números Fibonacci. Tente escrever o código para gerar os 20 primeiros números
na sequência. (Como uma dica, o vigésimo número Fibonacci é 6765.)
Desafio concluído
Veja um exemplo de solução analisando o código de exemplo finalizado no GitHub.
Com cada iteração do loop, você está pegando os últimos dois inteiros na lista, somando-os e adicionando esse
valor à lista. O loop será repetido até que você tenha adicionado 20 itens à lista.
Parabéns, você concluiu o tutorial de lista. Continue com o tutorial Introdução às classes em seu próprio
ambiente de desenvolvimento.
Saiba mais sobre como trabalhar com o tipo List no tópico Guia de .NET em coleções. Você também
aprenderá muitos outros tipos de coleção.
Explorar programação orientada a objeto com
classes e objetos
08/11/2019 • 15 minutes to read • Edit Online
Este tutorial espera que você tenha um computador que possa usar para desenvolvimento. O tutorial do .NET Olá,
mundo em 10 minutos tem instruções para configurar seu ambiente de desenvolvimento local no Windows, Linux
ou MacOS. Uma visão geral dos comandos que você usará está em Familiarize-se com as ferramentas de
desenvolvimento, com links para obter mais detalhes.
using System;
namespace classes
{
class Program
{
static void Main(string[] args)
{
Console.WriteLine("Hello World!");
}
}
}
Neste tutorial, você criará novos tipos que representam uma conta bancária. Normalmente, os desenvolvedores
definem cada classe em um arquivo de texto diferente. Isso facilita o gerenciamento à medida que o tamanho do
programa aumenta. Crie um novo arquivo chamado BankAccount.cs no diretório classes.
Esse arquivo conterá a definição de uma conta bancária. A programação Orientada a Objeto organiza o código
por meio da criação de tipos na forma de classes. Essas classes contêm o código que representa uma entidade
específica. A classe BankAccount representa uma conta bancária. O código implementa operações específicas por
meio de métodos e propriedades. Neste tutorial, a conta bancária dá suporte a este comportamento:
1. Ela tem um número com 10 dígitos que identifica exclusivamente a conta bancária.
2. Ela tem uma cadeia de caracteres que armazena o nome ou os nomes dos proprietários.
3. O saldo pode ser recuperado.
4. Ela aceita depósitos.
5. Ele aceita saques.
6. O saldo inicial deve ser positivo.
7. Os saques não podem resultar em um saldo negativo.
namespace classes
{
public class BankAccount
{
public string Number { get; }
public string Owner { get; set; }
public decimal Balance { get; }
Antes de continuar, vamos dar uma olhada no que você compilou. A declaração namespace fornece uma maneira
de organizar logicamente seu código. Este tutorial é relativamente pequeno, portanto, você colocará todo o código
em um namespace.
public class BankAccount define a classe ou o tipo que você está criando. Tudo dentro de { e } logo após a
declaração da classe define o comportamento da classe. Há cinco membros na classe BankAccount . As três
primeiras são propriedades. Propriedades são elementos de dados que podem ter um código que impõe a
validação ou outras regras. Os últimos dois são métodos. Os métodos são blocos de código que executam uma
única função. A leitura dos nomes de cada um dos membros deve fornecer informações suficientes para você, ou
outro desenvolvedor, entender o que a classe faz.
Construtores são chamados quando você cria um objeto usando new . Substitua a linha
Console.WriteLine("Hello World!"); em Program.cs pela linha a seguir (substitua <name> pelo seu nome):
Este é um membro de dados. Ele é private , o que significa que ele só pode ser acessado pelo código dentro da
classe BankAccount . É uma maneira de separar as responsabilidades públicas (como ter um número de conta) da
implementação privada (como os números de conta são gerados). Também é static , o que significa que ele é
compartilhado por todos os objetos BankAccount . O valor de uma variável não estática é exclusivo para cada
instância do objeto BankAccount . Adicione as duas linhas a seguir ao construtor para atribuir o número da conta:
this.Number = accountNumberSeed.ToString();
accountNumberSeed++;
using System;
namespace classes
{
public class Transaction
{
public decimal Amount { get; }
public DateTime Date { get; }
public string Notes { get; }
Agora, vamos adicionar um List<T> de Transaction objetos à classe BankAccount . Adicione a seguinte
declaração:
using System.Collections.Generic;
Agora, vamos alterar como Balance é reportado. Ele pode ser encontrado somando os valores de todas as
transações. Modifique a declaração do Balance na classe BankAccount para o seguinte:
return balance;
}
}
Este exemplo mostra um aspecto importante das propriedades. Agora, você está calculando o saldo quando outro
programador solicita o valor. Seu cálculo enumera todas as transações e fornece a soma como o saldo atual.
Depois, implemente os métodos MakeDeposit e MakeWithdrawal . Esses métodos aplicarão as duas últimas regras:
que o saldo inicial deve ser positivo, e que qualquer saque não pode criar um saldo negativo.
Isso introduz o conceito de exceções. A forma padrão de indicar que um método não pode concluir seu trabalho
com êxito é lançar uma exceção. O tipo de exceção e a mensagem associada a ele descrevem o erro. Aqui, o
método MakeDeposit lançará uma exceção se o valor do depósito for negativo. O método MakeWithdrawal lançará
uma exceção se o valor do saque for negativo ou se a aplicação do saque resultar em um saldo negativo:
A instrução throw lança uma exceção. A execução do bloco atual é encerrada e o controle transferido para o
bloco catch da primeira correspondência encontrado na pilha de chamadas. Você adicionará um bloco catch
para testar esse código um pouco mais tarde.
O construtor deve receber uma alteração para que adicione uma transação inicial, em vez de atualizar o saldo
diretamente. Como você já escreveu o método MakeDeposit , chame-o de seu construtor. O construtor concluído
deve ter esta aparência:
this.Owner = name;
MakeDeposit(initialBalance, DateTime.Now, "Initial balance");
}
DateTime.Now é uma propriedade que retorna a data e a hora atuais. Teste isso adicionando alguns depósitos e
saques ao método Main :
Depois, teste se você recebe condições de erro ao tentar criar uma conta com um saldo negativo:
Use as instruções try e catch para marcar um bloco de código que possa gerar exceções e detectar esses erros
esperados. Use a mesma técnica a fim de testar o código que gera uma exceção para um saldo negativo:
report.AppendLine("Date\tAmount\tNote");
foreach (var item in allTransactions)
{
report.AppendLine($"{item.Date.ToShortDateString()}\t{item.Amount}\t{item.Notes}");
}
return report.ToString();
}
Isso usa a classe StringBuilder para formatar uma cadeia de caracteres que contém uma linha para cada
transação. Você viu o código de formatação da cadeia de caracteres anteriormente nesses tutoriais. Um caractere
novo é \t . Ele insere uma guia para formatar a saída.
Adicione esta linha para testá-la no Program.cs:
Console.WriteLine(account.GetAccountHistory());
Próximas etapas
Se você ficou preso, pode ver a origem deste tutorial em nosso repositório GitHub.
Parabéns, você concluiu todos os nossos tutoriais de introdução ao C#. Se você estiver ansiosos para saber mais,
experimente mais nossos tutoriais.
Usar interpolação de cadeia de caracteres para
construir cadeia de caracteres formatadas
31/10/2019 • 15 minutes to read • Edit Online
Este tutorial ensina como usar a interpolação de cadeias de caracteres em C# para inserir valores em uma única
cadeia de caracteres de resultado. Escreva o código em C# e veja os resultados da compilação e da execução. O
tutorial contém uma série de lições que mostram como inserir valores em uma cadeia de caracteres e formatar
esses valores de diferentes maneiras.
Este tutorial espera que você tenha um computador que possa usar para desenvolvimento. O tutorial do .NET Olá,
mundo em 10 minutos tem instruções para configurar seu ambiente de desenvolvimento local no Windows, Linux
ou MacOS. Você também pode concluir a versão interativa deste tutorial em seu navegador.
Esse comando cria um novo aplicativo de console .NET Core no diretório atual.
Abra Program.cs em seu editor favorito e substitua a linha Console.WriteLine("Hello World!"); pelo seguinte
código, em que você substitui <name> pelo seu nome:
Experimente este código digitando dotnet run na sua janela do console. Ao executar o programa, ele exibe uma
única cadeia de caracteres que inclui seu nome na saudação. A cadeia de caracteres incluída na chamada de
método WriteLine é uma expressão de cadeia de caracteres interpolada. Ela é um tipo de modelo que permite que
você construa uma única cadeia de caracteres (chamado de cadeia de caracteres de resultado) com base em uma
cadeia de caracteres que inclui o código inserido. As cadeias de caracteres interpoladas são particularmente úteis
para inserir valores em uma cadeia de caracteres ou para concatenar (unir) cadeias de caracteres.
Esse exemplo simples contém os dois elementos que toda cadeia de caracteres interpolada deve ter:
Um literal de cadeia de caracteres que começa com o caractere $ antes do caractere de aspas de abertura.
Não pode haver nenhum espaço entre o símbolo $ e o caractere de aspas. (Se você quiser ver o que
acontece ao incluir um espaço, insira um após o caractere $ , salve o arquivo e execute novamente o
programa, digitando dotnet run na janela do console. O compilador do C# exibirá uma mensagem de erro
"Erro CS1056: caractere '$' inesperado".)
Uma ou mais expressões de interpolação. Uma expressão de interpolação é indicada por chaves de abertura
e fechamento ( { e } ). Você pode colocar qualquer expressão de C# que retorne um valor (incluindo
null ) dentro das chaves.
Vamos testar mais alguns exemplos de interpolação de cadeias de caracteres com outros tipos de dados.
Incluir diferentes tipos de dados
Na seção anterior, você usou a interpolação de cadeias de caracteres para inserir uma cadeia de caracteres dentro
de outra. Entretanto, o resultado de uma expressão de interpolação pode ser de qualquer tipo de dados. Vamos
incluir valores de vários tipos de dados em uma cadeia de caracteres interpolada.
No exemplo a seguir, primeiramente definimos um tipo de dados de classe Vegetable que tem uma propriedade
Name e um método ToString , que substitui o comportamento do método Object.ToString(). O modificador de
acesso public disponibiliza esse método para qualquer código de cliente para obter a representação de cadeia de
caracteres de uma instância de Vegetable . No exemplo, o método Vegetable.ToString retorna o valor da
propriedade Name que é inicializada no construtor Vegetable :
Em seguida, criamos uma instância da classe Vegetable chamada item usando o new operador e fornecendo
um nome para o construtor Vegetable :
Por fim, incluímos a variável item em uma cadeia de caracteres interpolada que também contém um valor
DateTime, um valor Decimal e um valor de enumeração Unit valor. Substitua todo o código C# em seu editor
pelo seguinte código e, depois, use o comando dotnet run para executá-lo:
using System;
Observe que a expressão de interpolação item na cadeia de caracteres interpolada é resolvida como o texto
"eggplant" na cadeia de caracteres de resultado. Isso ocorre porque, quando o tipo do resultado da expressão não
é uma cadeia de caracteres, o resultado é resolvido como uma cadeia de caracteres da seguinte maneira:
Se a expressão de interpolação for avaliada como null , uma cadeia de caracteres vazia ("" ou
String.Empty) será usada.
Se a expressão de interpolação não foi avaliada como null , normalmente o método ToString do tipo de
resultado será chamado. Você pode testar isso atualizando a implementação do método
Vegetable.ToString . Talvez nem seja necessário implementar o método ToString , pois cada tipo tem
algum modo de implementação desse método. Para testar isso, comente a definição do método
Vegetable.ToString no exemplo (para isso, coloque o símbolo de comentário // na frente dele). Na saída,
a cadeia de caracteres "eggplant" é substituída pelo nome do tipo totalmente qualificado, ("Vegetable" neste
exemplo), que é o comportamento padrão do método Object.ToString(). O comportamento padrão do
método ToString para um valor de enumeração é retornar a representação de cadeia de caracteres do
valor.
Na saída deste exemplo, a data é muito precisa (o preço de "eggplant" não muda a cada segundo) e o valor do
preço não indica uma unidade monetária. Na próxima seção, você aprenderá como corrigir esses problemas
controlando o formato das representações das cadeias de caracteres dos resultados de expressão.
Você especifica uma cadeia de caracteres de formato colocando dois-pontos (":") e a cadeia de caracteres de
formato após a expressão de interpolação. "d" é uma cadeia de caracteres de formato de data e hora padrão que
representa o formato de data abreviada. "C2" é um cadeia de caracteres de formato numérico padrão que
representa um número como um valor de moeda com dois dígitos após o ponto decimal.
Diversos tipos nas bibliotecas do .NET são compatíveis com um conjunto predefinido de cadeias de caracteres de
formato. Isso inclui todos os tipos numéricos e os tipos de data e hora. Para obter uma lista completa dos tipos
que são compatíveis com as cadeias de caracteres de formato, consulte Cadeias de caracteres de formato e tipos
da biblioteca de classes do .NET no artigo Tipos de formatação no .NET.
Tente modificar as cadeias de caracteres de formato em seu editor de texto e, sempre que fizer uma alteração,
execute novamente o programa para ver como as alterações afetam a formatação da data e hora e do valor
numérico. Altere o "d" em {date:d} para "t" (para exibir o formato de hora abreviada), para "y" (para exibir o ano
e o mês) e para "yyyy" (para exibir o ano como um número de quatro dígitos). Altere o "C2" em {price:C2} para
"e" (para obter notação exponencial) e para "F3" (para um valor numérico com três dígitos após o ponto decimal).
Além de controlar a formatação, você também pode controlar a largura do campo e o alinhamento das cadeias de
caracteres formatadas incluídas na cadeia de caracteres de resultado. Na próxima seção, você aprenderá como
fazer isso.
Os nomes de autores são alinhados à esquerda e os títulos que eles escreveram são alinhados à direita. Você
especifica o alinhamento adicionando uma vírgula (",") após a expressão de interpolação e designando a largura
mínima do campo. Se o valor especificado for um número positivo, o campo será alinhado à direita. Se for um
número negativo, o campo será alinhado à esquerda.
Tente remover os sinais negativos do código {"Author",-25} e {title.Key,-25} e execute o exemplo novamente,
como feito no código a seguir:
Console.WriteLine($"|{"Author",25}|{"Title",30}|");
foreach (var title in titles)
Console.WriteLine($"|{title.Key,25}|{title.Value,30}|");
Este tutorial mostra como usar a interpolação de cadeia de caracteres para formatar e incluir resultados de
expressão em uma cadeia de caracteres de resultado. Os exemplos pressupõem que você esteja familiarizado
com os conceitos básicos do C# e a formatação de tipos do .NET. Se você não estiver familiarizado com a
interpolação de cadeia de caracteres ou com a formatação de tipos do .NET, confira primeiro o tutorial interativo
sobre a interpolação de cadeia de caracteres. Para obter mais informações sobre como formatar tipos no .NET,
confira o tópico Formatando tipos no .NET.
NOTE
Os exemplos de C# neste artigo são executados no executador de código embutido Try.NET e no playground. Clique no
botão Executar para executar um exemplo em uma janela interativa. Ao executar o código, é possível modificá-lo e executar
o código modificado clicando em Executar novamente. O código modificado será executado na janela interativa ou, se a
compilação falhar, a janela interativa exibirá todos as mensagens de erro do compilador C#.
Introdução
O recurso interpolação de cadeia de caracteres baseia-se no recurso formatação composta e fornece uma sintaxe
mais legível e conveniente para incluir resultados de expressão formatada em uma cadeia de caracteres de
resultado.
Para identificar uma literal de cadeia de caracteres como uma cadeia de caracteres interpolada, preceda-a com o
símbolo $ . Você pode inserir qualquer expressão C# válida que retorna um valor em uma cadeia de caracteres
interpolada. No seguinte exemplo, assim que uma expressão é avaliada, o resultado é convertido em uma cadeia
de caracteres e incluído em uma cadeia de caracteres de resultado:
double a = 3;
double b = 4;
Console.WriteLine($"Area of the right triangle with legs of {a} and {b} is {0.5 * a * b}");
Console.WriteLine($"Length of the hypotenuse of the right triangle with legs of {a} and {b} is
{CalculateHypotenuse(a, b)}");
double CalculateHypotenuse(double leg1, double leg2) => Math.Sqrt(leg1 * leg1 + leg2 * leg2);
// Expected output:
// Area of the right triangle with legs of 3 and 4 is 6
// Length of the hypotenuse of the right triangle with legs of 3 and 4 is 5
Como mostra o exemplo, você inclui uma expressão em uma cadeia de caracteres interpolada colocando-a com
chaves:
{<interpolationExpression>}
Cadeia de caracteres interpoladas são compatíveis com todos os recursos do recurso formatação composta de
cadeia de caracteres. Isso as torna uma alternativa mais legível ao uso do método String.Format.
{<interpolationExpression>:<formatString>}
O seguinte exemplo mostra como especificar cadeias de caracteres de formato padrão e personalizadas para
expressões que produzem resultados numéricos ou de data e hora:
// Expected output:
// On Sunday, November 25, 1731 Leonhard Euler introduced the letter e to denote 2.71828 in a letter to
Christian Goldbach.
Para obter mais informações, consulte a seção Componente de cadeia de caracteres de formato do tópico
Formatação composta. Esta seção fornece links para tópicos que descrevem cadeias de caracteres de formatos
padrão e personalizado compatíveis com os tipos base do .NET.
{<interpolationExpression>,<alignment>}
Se o valor alignment for positivo, o resultado da expressão formatada será alinhado à direita; se for negativo, ele
será alinhado à esquerda.
Caso precise especificar o alinhamento e uma cadeia de caracteres de formato, comece com o componente de
alinhamento:
{<interpolationExpression>,<alignment>:<formatString>}
O seguinte exemplo mostra como especificar o alinhamento e usa caracteres de barra vertical ("|") para delimitar
campos de texto:
double a = 3;
double b = 4;
Console.WriteLine($"Three classical Pythagorean means of {a} and {b}:");
Console.WriteLine($"|{"Arithmetic",NameAlignment}|{0.5 * (a + b),ValueAlignment:F3}|");
Console.WriteLine($"|{"Geometric",NameAlignment}|{Math.Sqrt(a * b),ValueAlignment:F3}|");
Console.WriteLine($"|{"Harmonic",NameAlignment}|{2 / (1 / a + 1 / b),ValueAlignment:F3}|");
// Expected output:
// Three classical Pythagorean means of 3 and 4:
// |Arithmetic| 3.500|
// |Geometric| 3.464|
// |Harmonic | 3.429|
Como mostra a saída de exemplo, se o tamanho do resultado da expressão formatada exceder a largura de
campo especificada, o valor alignment será ignorado.
Para obter mais informações, consulte a seção Componente de alinhamento do tópico Formatação composta.
// Expected output:
// Find the intersection of the {1, 2, 7, 9} and {7, 9, 12} sets.
// C:\Users\Jane\Documents
// C:\Users\Jane\Documents
Como mostra o exemplo, você pode usar uma instância FormattableString para gerar várias cadeias de caracteres
de resultado para várias culturas.
Conclusão
Este tutorial descreve cenários comuns de uso da interpolação de cadeia de caracteres. Para obter mais
informações sobre a interpolação de cadeia de caracteres, consulte o tópico Interpolação de cadeia de caracteres.
Para obter mais informações sobre como formatar tipos no .NET, confira os tópicos Formatando tipos no .NET e
Formatação composta.
Consulte também
String.Format
System.FormattableString
System.IFormattable
Cadeias de Caracteres
Tutorial: atualizar interfaces com métodos de
interface padrão C# no 8,0
30/10/2019 • 11 minutes to read • Edit Online
Desde o C# 8.0 no .NET Core 3.0, é possível definir uma implementação em que você declara um membro de uma
interface. O cenário mais comum é adicionar membros com segurança a uma interface já lançada e usada por
vários clientes.
Neste tutorial, você aprenderá a:
Estender interfaces com segurança, adicionando métodos com implementações.
Criar implementações parametrizadas para oferecer maior flexibilidade.
Permitir que os implementadores forneçam uma implementação mais específica na forma de uma substituição.
Prerequisites
Você precisará configurar seu computador para executar o .NET Core, incluindo o C# compilador 8,0. O C#
compilador 8,0 está disponível a partir do Visual Studio 2019 versão 16,3 ou do SDK do .NET Core 3,0.
Dessas interfaces, a equipe poderia criar uma biblioteca para os usuários criarem uma experiência melhor para os
clientes. A meta era criar um relacionamento mais profundo com os clientes existentes e melhorar o
relacionamento com clientes novos.
Agora é hora de atualizar a biblioteca para a próxima versão. Um dos recursos solicitados habilitará um desconto
de fidelidade para os clientes que tiverem muitos pedidos. Esse novo desconto de fidelidade é aplicado sempre que
um cliente faz um pedido. O desconto específico é uma propriedade de cada cliente. Cada implementação de
ICustomer pode definir regras diferentes para o desconto de fidelidade.
A forma mais natural de adicionar essa funcionalidade é melhorar a interface ICustomer com um método para
aplicar qualquer desconto de fidelidade. Essa sugestão de design causou uma preocupação entre os
desenvolvedores experientes: "as interfaces são imutáveis depois de terem sido lançadas! Esta é uma alteração da
falha!" O C# 8.0 adiciona implementações de interface padrão para interfaces de atualização. Os autores de
biblioteca podem adicionar novos membros à interface e fornecer uma implementação padrão para esses
membros.
Implementações de interface padrão permitem que os desenvolvedores atualizem uma interface, permitindo que
qualquer implementador substitua essa implementação. Os usuários da biblioteca podem aceitar a implementação
padrão como uma alteração da falha. Se as regras de negócio forem diferentes, elas poderão substituir a
implementação.
// Version 1:
public decimal ComputeLoyaltyDiscount()
{
DateTime TwoYearsAgo = DateTime.Now.AddYears(-2);
if ((DateJoined < TwoYearsAgo) && (PreviousOrders.Count() > 10))
{
return 0.10m;
}
return 0;
}
Essa conversão de SampleCustomer em ICustomer é necessária. A classe SampleCustomer não precisa fornecer uma
implementação de ComputeLoyaltyDiscount ; isso é fornecido pela interface ICustomer . No entanto, a classe
SampleCustomer não herda membros de suas interfaces. Essa regra não foi alterada. Para chamar qualquer método
declarado e implementado na interface, a variável deve ser o tipo da interface: ICustomer neste exemplo.
Fornecer parametrização
Esse é um bom início. Porém, a implementação padrão é restritiva demais. Muitos consumidores desse sistema
podem escolher limites diferentes para o número de compras, um período diferente de associação ou um
percentual de desconto diferente. Você pode fornecer uma experiência de atualização melhor para mais clientes,
fornecendo uma maneira de definir esses parâmetros. Vamos adicionar um método estático que defina esses três
parâmetros, controlando a implementação padrão:
// Version 2:
public static void SetLoyaltyThresholds(
TimeSpan ago,
int minimumOrders = 10,
decimal percentageDiscount = 0.10m)
{
length = ago;
orderCount = minimumOrders;
discountPercent = percentageDiscount;
}
private static TimeSpan length = new TimeSpan(365 * 2, 0,0,0); // two years
private static int orderCount = 10;
private static decimal discountPercent = 0.10m;
Há muitos novos recursos de linguagem mostrados no fragmento de código pequeno. Agora as interfaces podem
incluir membros estáticos, incluindo campos e métodos. Modificadores de acesso diferentes também estão
habilitados. Os campos adicionais são particulares, o novo método é público. Qualquer dos modificadores são
permitidos em membros de interface.
Aplicativos que usam a fórmula geral para calcular o desconto de fidelidade, mas que usam parâmetros diferentes,
não precisam fornecer uma implementação personalizada; eles podem definir os argumentos por meio de um
método estático. Por exemplo, o código a seguir define um "agradecimento ao cliente" que recompensa qualquer
cliente com mais de um mês de associação:
Em uma implementação de uma classe que implementa essa interface, a substituição pode chamar o método
auxiliar estático e estender essa lógica para fornecer o desconto de "novo cliente":
Você pode ver todo o código concluído no nosso repositório de amostras no GitHub. Você pode obter o aplicativo
de iniciante em nosso repositório de exemplos no GitHub.
Esses novos recursos significam que interfaces podem ser atualizadas com segurança quando há uma
implementação padrão razoável para os novos membros. Projete interfaces cuidadosamente para expressar ideias
funcionais únicas que possam ser implementadas por várias classes. Isso torna mais fácil atualizar essas definições
de interface quando são descobertos novos requisitos para a mesma ideia funcional.
Tutorial: misturar funcionalidade no ao criar classes
usando interfaces com métodos de interface padrão
25/11/2019 • 15 minutes to read • Edit Online
Desde o C# 8.0 no .NET Core 3.0, é possível definir uma implementação em que você declara um membro de uma
interface. Esse recurso fornece novos recursos em que você pode definir implementações padrão para recursos
declarados em interfaces. As classes podem escolher quando substituir a funcionalidade, quando usar a
funcionalidade padrão e quando não declarar suporte para recursos discretos.
Neste tutorial, você aprenderá a:
Crie interfaces com implementações que descrevem recursos discretos.
Crie classes que usam as implementações padrão.
Crie classes que substituem algumas ou todas as implementações padrão.
Prerequisites
Você precisará configurar seu computador para executar o .NET Core, incluindo o C# compilador 8,0. O C#
compilador 8,0 está disponível a partir do Visual Studio 2019, 16,3ou do SDK do .NET Core 3,0 ou posterior.
Criar o aplicativo
Considere um aplicativo de automação inicial. Você provavelmente tem muitos tipos diferentes de luzes e
indicadores que poderiam ser usados em toda a casa. Cada luz deve dar suporte a APIs para ligá-las e reportar o
estado atual. Algumas luzes e indicadores podem dar suporte a outros recursos, como:
Ative a luz e desative-a após um temporizador.
Pisque a luz por um período de tempo.
Alguns desses recursos estendidos podem ser emulados em dispositivos que dão suporte ao conjunto mínimo.
Isso indica fornecer uma implementação padrão. Para os dispositivos que têm mais recursos internos, o software
do dispositivo usaria os recursos nativos. Para outras luzes, eles poderiam optar por implementar a interface e usar
a implementação padrão.
Membros de interface padrão é uma solução melhor para esse cenário do que métodos de extensão. Os autores de
classe podem controlar quais interfaces eles optam por implementar. Essas interfaces que eles escolhem estão
disponíveis como métodos. Além disso, como os métodos de interface padrão são virtuais por padrão, a expedição
do método sempre escolhe a implementação na classe.
Vamos criar o código para demonstrar essas diferenças.
Criar interfaces
Comece criando a interface que define o comportamento para todas as luzes:
Um acessório de luz de sobrecarga básica pode implementar essa interface, conforme mostrado no código a
seguir:
public override string ToString() => $"The light is {isOn: \"on\", \"off\"}";
Neste tutorial, o código não impulsiona os dispositivos IoT, mas emula essas atividades gravando mensagens no
console. Você pode explorar o código sem automatizar sua casa.
Em seguida, vamos definir a interface para uma luz que pode desligar automaticamente após um tempo limite:
Você pode adicionar uma implementação básica à luz de sobrecarga, mas uma solução melhor é modificar essa
definição de interface para fornecer uma virtual implementação padrão:
public interface ITimerLight : ILight
{
public async Task TurnOnFor(int duration)
{
Console.WriteLine("Using the default interface method for the ITimerLight.TurnOnFor.");
SwitchOn();
await Task.Delay(duration);
SwitchOff();
Console.WriteLine("Completed ITimerLight.TurnOnFor sequence.");
}
}
Ao adicionar essa alteração, a classe OverheadLight pode implementar a função de temporizador declarando
suporte para a interface:
Um tipo de luz diferente pode dar suporte a um protocolo mais sofisticado. Ele pode fornecer sua própria
implementação para TurnOnFor , conforme mostrado no código a seguir:
Ao contrário da substituição de métodos de classe virtual, a declaração de TurnOnFor na classe HalogenLight não
usa a palavra-chave override .
A implementação padrão permite que qualquer luz pisque. A luz de sobrecarga pode adicionar recursos de
temporizador e de intermitência usando a implementação padrão:
public override string ToString() => $"The light is {isOn: \"on\", \"off\"}";
}
Um novo tipo de luz, o LEDLight dá suporte à função de temporizador e à função de intermitência diretamente.
Esse estilo leve implementa as interfaces ITimerLight e IBlinkingLight e substitui o método Blink :
public override string ToString() => $"The light is {isOn: \"on\", \"off\"}";
}
public override string ToString() => $"The light is {isOn: \"on\", \"off\"}";
}
O HalogenLight criado anteriormente não dá suporte a intermitência. Portanto, não adicione o IBlinkingLight à
lista de suas interfaces com suporte.
Essas alterações são compiladas corretamente, mesmo que o ExtraFancyLight declare o suporte para a interface
ILight e as interfaces derivadas ITimerLight e IBlinkingLight . Há apenas uma implementação "mais próxima"
declarada na interface ILight . Qualquer classe que declarou uma substituição se tornaria a única implementação
"mais próxima". Você viu exemplos nas classes anteriores que substituiu os membros de outras interfaces
derivadas.
Evite substituir o mesmo método em várias interfaces derivadas. Isso cria uma chamada de método ambígua
sempre que uma classe implementa ambas as interfaces derivadas. O compilador não pode escolher um único
método melhor para que ele emita um erro. Por exemplo, se o IBlinkingLight e ITimerLight implementarem
uma substituição de PowerStatus , o OverheadLight precisaria fornecer uma substituição mais específica. Caso
contrário, o compilador não pode escolher entre as implementações nas duas interfaces derivadas. Normalmente,
você pode evitar essa situação mantendo as definições de interface pequenas e concentradas em um recurso.
Nesse cenário, cada recurso de uma luz é sua própria interface; várias interfaces são herdadas apenas por classes.
Este exemplo mostra um cenário em que você pode definir recursos discretos que podem ser misturados em
classes. Você declara qualquer conjunto de funcionalidades com suporte declarando a quais interfaces uma classe
dá suporte. O uso de métodos de interface padrão virtuais permite que as classes usem ou definam uma
implementação diferente para qualquer ou todos os métodos de interface. Esse recurso de linguagem fornece
novas maneiras de modelar os sistemas do mundo real que você está criando. Os métodos de interface padrão
fornecem uma maneira mais clara de expressar classes relacionadas que podem misturar e corresponder recursos
diferentes usando implementações virtuais desses recursos.
Índices e intervalos
30/10/2019 • 7 minutes to read • Edit Online
Intervalos e índices fornecem uma sintaxe sucinta para acessar elementos únicos ou intervalos em uma
sequência.
Neste tutorial, você aprenderá a:
Use a sintaxe para intervalos em uma sequência.
Compreenda as decisões de design para o início e o fim de cada sequência.
Conheça cenários para os tipos Index e Range.
Você pode recuperar a última palavra com o índice ^1 . Adicione o código a seguir abaixo da inicialização:
Um intervalo especifica o início e o final de um intervalo. Intervalos são exclusivos, o que significa que final não
está incluído no intervalo. O intervalo [0..^0] representa todo o intervalo, assim como [0..sequence.Length]
representa todo o intervalo.
O código a seguir cria um subintervalo com as palavras "quick", "brown" e "fox". Ele inclui words[1] até words[3]
. O elemento words[4] não está no intervalo. Adicione o seguinte código ao mesmo método. Copie e cole-o na
parte inferior da janela interativa.
string[] quickBrownFox = words[1..4];
foreach (var word in quickBrownFox)
Console.Write($"< {word} >");
Console.WriteLine();
O código a seguir cria um subintervalo com "lazy" e "dog". Ele inclui words[^2] e words[^1] . O índice final
words[^0] não está incluído. Adicione o seguinte código também:
Você também pode declarar intervalos ou índices como variáveis. A variável então pode ser usada dentro dos
caracteres [ e ] :
O exemplo a seguir mostra muitos dos motivos para essas escolhas. Modifique x , y e z para tentar
combinações diferentes. Quando você testar, use valores em que x é menor que y e y é menor que z para as
combinações válidas. Adicione o seguinte código a um novo método. Tente usar combinações diferentes:
int[] numbers = Enumerable.Range(0, 100).ToArray();
int x = 12;
int y = 25;
int z = 36;
(int min, int max, double average) MovingAverage(int[] subSequence, Range range) =>
(
subSequence[range].Min(),
subSequence[range].Max(),
subSequence[range].Average()
);
C#8,0 apresenta tipos de referência anuláveis, que complementam os tipos de referência da mesma maneira que
os tipos de valores de valor nulo complementam tipos Value. Para declarar que uma variável é um tipo de
referência que permite valor nulo, anexe um ? ao tipo. Por exemplo, string? representa uma string que
permite valor nulo. Você pode usar esses novos tipos para expressar mais claramente sua intenção de design:
algumas variáveis devem sempre ter um valor, outras podem ter um valor ausente.
Neste tutorial, você aprenderá a:
Incorporar tipos de referência que permitem valores nulos e tipos de referência que não permitem valores
nulos aos designs
Habilitar verificações de tipo de referência que permitem valor nulo em todo o código.
Gravar código em locais onde o compilador imponha essas decisões de design.
Usar o recurso de referência que permite valor nulo em seus próprios designs
Prerequisites
Você precisará configurar seu computador para executar o .NET Core, incluindo o C# compilador 8,0. O C#
compilador 8,0 está disponível com o Visual Studio 2019ou o .NET Core 3,0.
Este tutorial pressupõe que você esteja familiarizado com o C# e .NET, incluindo o Visual Studio ou a CLI do .NET
Core.
Criar a pesquisa com tipos que permitem valor nulo e tipos que não
permitem valor nulo
O primeiro código gravado criará a pesquisa. Você escreverá classes para modelar uma pergunta da pesquisa e
uma execução da pesquisa. A pesquisa tem três tipos de perguntas, diferenciadas pelo formato da resposta:
respostas do tipo Sim/Não, respostas com números e respostas com texto. Criar uma classe de
public SurveyQuestion :
namespace NullableIntroduction
{
public class SurveyQuestion
{
}
}
O compilador interpreta cada declaração de variável de tipo de referência como um tipo de referência não
anulável para o código em um contexto de anotação anulável habilitado. Para ver seu primeiro aviso, adicione
propriedades ao texto da pergunta e tipo de pergunta, conforme mostrado no código a seguir:
namespace NullableIntroduction
{
public enum QuestionType
{
YesNo,
Number,
Text
}
Como você não inicializou QuestionText , o compilador emitirá um aviso informando que uma propriedade que
não permite valor nulo não foi inicializada. Seu design exige que o texto da pergunta não seja um valor nulo,
portanto, você inclui um construtor para inicializá-lo e o valor QuestionType também. A definição da classe
concluída se parece com o código a seguir:
namespace NullableIntroduction
{
public enum QuestionType
{
YesNo,
Number,
Text
}
A adição do construtor removerá o aviso. O argumento do construtor também é um tipo de referência que não
permite valor nulo, portanto, o compilador não emite avisos.
Em seguida, crie uma classe public chamada SurveyRun . Esta classe contém uma lista de métodos e objetos
SurveyQuestion para adicionar perguntas à pesquisa, conforme mostrado no código a seguir:
using System.Collections.Generic;
namespace NullableIntroduction
{
public class SurveyRun
{
private List<SurveyQuestion> surveyQuestions = new List<SurveyQuestion>();
Como foi feito anteriormente, você deve inicializar o objeto de lista com um valor não nulo ou o compilador
emitirá um aviso. Não há verificações de valores nulos na segunda sobrecarga de AddQuestion , pois elas são
desnecessárias: você declarou que a variável não permite valor nulo. Seu valor não pode ser null .
Alterne para Program.cs no seu editor e substitua o conteúdo de Main pelas seguintes linhas de código:
Como o projeto inteiro está em um contexto de anotação anulável habilitado, você receberá avisos quando passar
null para qualquer método que espera um tipo de referência não anulável. Experimente adicionar a seguinte
linha a Main :
surveyRun.AddQuestion(QuestionType.Text, default);
namespace NullableIntroduction
{
public class SurveyResponse
{
public int Id { get; }
A principal responsabilidade dessa classe é gerar as respostas de um participante para as perguntas da pesquisa.
Essa responsabilidade conta com algumas etapas:
1. Peça para participar da pesquisa. Se a pessoa não consentir, retorne uma resposta de ausente (ou de valor
nulo).
2. Faça as perguntas e registre a resposta. As respostas também pode ser ausentes (ou de valor nulo).
Adicione o seguinte código à classe SurveyResponse :
Aqui, novamente, sua opção por uma List<SurveyResponse>? que permite valor nulo indica que a resposta pode
ser um valor nulo. Isso indica que a pesquisa ainda não foi entregue a nenhum pesquisado. Observe que os
entrevistados são adicionados até que um suficiente de pessoas tiver consentido.
A última etapa para executar a pesquisa é adicionar uma chamada para executar a pesquisa no final do método
Main :
surveyRun.PerformSurvey(50);
Como surveyResponses é um tipo de referência anulável, são necessárias verificações nulas antes de fazer
referência a ela. O método Answer retorna uma cadeia de caracteres não anulável, portanto, precisamos abordar
o caso de uma resposta ausente usando o operador de União nulo.
Em seguida, adicione esses três membros com corpo de expressão à classe SurveyRun :
public IEnumerable<SurveyResponse> AllParticipants => (respondents ?? Enumerable.Empty<SurveyResponse>());
public ICollection<SurveyQuestion> Questions => surveyQuestions;
public SurveyQuestion GetQuestion(int index) => surveyQuestions[index];
O membro AllParticipants deve levar em conta que a variável respondents pode ser um valor nulo, mas o valor
de retorno não pode ser nulo. Se você alterar essa expressão removendo ?? e a sequência vazia que se segue, o
compilador avisará que o método poderá retornar null e sua assinatura de retorno retornará um tipo que não
permite valor nulo.
Por fim, adicione o seguinte loop à parte inferior do método Main :
Você não precisa de verificações de null neste código porque criou as interfaces subjacentes para que todas elas
retornem tipos de referência que não permitem valor nulo.
Obter o código
Obtenha o código do tutorial concluído em nosso repositório de amostras na pasta csharp/NullableIntroduction.
Experimente alterar as declarações de tipo entre os tipos de referência que permitem valor nulo e tipos de
referência que não permitem valor nulo. Veja como isso gera avisos diferentes para garantir que um null não
será acidentalmente cancelado.
Próximas etapas
Saiba mais migrando um aplicativo existente para usar tipos de referência anuláveis:
Atualizar um aplicativo a fim de usar tipos de referência anuláveis
Tutorial: migrar código existente com tipos de
referência anuláveis
25/11/2019 • 25 minutes to read • Edit Online
O C# 8 introduz tipos de referência que permitem valor nulo que complementam os tipos de referência da
mesma maneira que os tipos de valor que permitem valor nulo complementam os tipos de valor. Para declarar
que uma variável é um tipo de referência que permite valor nulo, anexe um ? ao tipo. Por exemplo, string?
representa uma string que permite valor nulo. Você pode usar esses novos tipos para expressar mais
claramente sua intenção de design: algumas variáveis devem sempre ter um valor, outras podem ter um valor
ausente. Quaisquer variáveis existentes de um tipo de referência seriam interpretadas como um tipo de referência
não anulável.
Neste tutorial, você aprenderá a:
Habilitar verificações de referência nula enquanto trabalha com o código.
Diagnosticar e corrigir avisos diferentes relacionados a valores nulos.
Gerenciar a interface entre contextos habilitados para anulável e desabilitados para anulável.
Controlar contextos de anotação anuláveis.
Prerequisites
Você precisará configurar seu computador para executar o .NET Core, incluindo o C# compilador 8,0. O C#
compilador 8 está disponível a partir do Visual Studio 2019 versão 16,3 ou do SDK do .NET Core 3,0.
Este tutorial pressupõe que você esteja familiarizado com o C# e .NET, incluindo o Visual Studio ou a CLI do .NET
Core.
A atualização da versão da linguagem seleciona C# 8.0, mas não habilita o contexto de anotação anulável ou o
contexto de aviso anulável. Recompile o projeto para garantir que ocorra sem avisos.
Em seguida, ative o contexto de anotação anulável e veja quantos avisos são gerados. Adicione o seguinte
elemento aos dois arquivos csproj na solução, diretamente abaixo do elemento LangVersion :
<Nullable>enable</Nullable>
Faça uma compilação de teste e observe a lista de avisos. Neste aplicativo pequeno, o compilador gera cinco
avisos. Provavelmente, você deixaria o contexto de anotação anulável habilitado e iniciaria a correção de avisos de
todo o projeto.
Essa estratégia funciona apenas para projetos menores. Para os projetos maiores, o número de avisos gerados
com a habilitação do contexto de anotação anulável em toda a base de código dificulta a correção sistemática dos
avisos. Para projetos corporativos maiores, convém migrar um projeto de cada vez. Em cada projeto, migre uma
classe ou arquivo por vez.
#nullable enable
public class NewsStoryViewModel
{
public DateTimeOffset Published { get; set; }
public string Title { get; set; }
public string Uri { get; set; }
}
#nullable restore
Essas duas diretivas ajudam você a concentrar seus esforços de migração. Os avisos anuláveis são gerados para a
área do código na qual você está trabalhando ativamente. Deixe-os ativos até que você esteja pronto para ativar
os avisos de todo o projeto. Use restore em vez do valor disable para não desabilitar acidentalmente o
contexto após ativar as anotações anuláveis do projeto inteiro. Depois de ativar o contexto de anotação anulável
para de todo o projeto, remova todas as pragmas #nullable desse projeto.
A classe NewsStoryViewModel é um DTO (objeto de transferência de dados) e duas propriedades são cadeias de
caracteres de leitura/gravação:
Essas duas propriedades causam CS8618 , "Propriedade não anulável não inicializada". Está bem claro: as duas
propriedades string têm o valor padrão de null quando um NewsStoryViewModel é construído. O importante é
descobrir como os objetos NewsStoryViewModel são construídos. Examinando essa classe, não é possível
determinar se o valor null faz parte do design, ou se esses objetos são definidos como valores não nulos
sempre que um é criado. As histórias de notícias são criadas no método GetNews da classe NewsService :
Há muita coisa acontecendo no bloco de código anterior. Esse aplicativo usa o pacote NuGet AutoMapper para
construir um item de notícias a partir de um ISyndicationItem . Você descobriu que os itens de notícias são
construídos, e as propriedades são definidas nessa instrução. Isso significa que o design do NewsStoryViewModel
indica que essas propriedades nunca devem ter o valor null . Essas propriedades devem ser tipos de referência
não anuláveis. Isso expressa melhor a intenção do design original. Na verdade, qualquer NewsStoryViewModel é
corretamente instanciada com valores não nulos. Isso torna o código de inicialização a seguir uma correção válida:
A atribuição de Title e Uri a default , que é null para o tipo string , não altera o comportamento de
runtime do programa. O NewsStoryViewModel ainda é construído com valores nulos, mas agora o compilador não
retorna avisos. O operador que tolera valores nulos, o caractere ! logo após a expressão default informa ao
compilador que a expressão anterior não é nula. Essa técnica pode ser adequada quando outras alterações forçam
alterações muito maiores em uma base de código, mas nesse aplicativo há uma solução relativamente rápida e
melhor: tornar o NewsStoryViewModel um tipo imutável em que todas as propriedades são definidas no construtor.
Faça estas alterações em NewsStoryViewModel :
#nullable enable
public class NewsStoryViewModel
{
public NewsStoryViewModel(DateTimeOffset published, string title, string uri) =>
(Published, Title, Uri) = (published, title, uri);
Depois disso, atualize o código que configura o AutoMapper, para que ele use o construtor em vez de definir
propriedades. Abra NewsService.cs e procure o seguinte código na parte inferior do arquivo:
#nullable enable
public class NewsStoryProfile : Profile
{
public NewsStoryProfile()
{
// Create the AutoMapper mapping profile between the 2 objects.
// ISyndicationItem.Id maps to NewsStoryViewModel.Uri.
CreateMap<ISyndicationItem, NewsStoryViewModel>()
.ForCtorParam("uri", opt => opt.MapFrom(src => src.Id));
}
Note que como essa classe é pequena, e você examinou cuidadosamente, ative a diretiva #nullable enable acima
dessa declaração de classe. A alteração no construtor poderia ter quebrado algo, portanto, vale a pena executar
todos os testes e testar o aplicativo antes de prosseguir.
O primeiro conjunto de alterações mostrou como descobrir quando o design original indicou que as variáveis não
devem ser definidas como null . A técnica é conhecida como corrigir pela construção. Você declara que um
objeto e suas propriedades não podem ser null quando ele é construído. A análise de fluxo do compilador
fornece garantia de que essas propriedades não são definidas como null após a construção. Observe que esse
construtor é chamado pelo código externo, e que o código é indiferente ao anulável. A nova sintaxe não
fornece uma verificação de runtime. O código externo pode enganar a análise de fluxo do compilador.
Outras vezes, a estrutura de uma classe fornece dicas sobre diferentes sobre a intenção. Abra o arquivo
Error.cshtml.cs na pasta Páginas. O ErrorViewModel contém o seguinte código:
Adicione a diretiva #nullable enable antes da declaração de classe e uma diretiva #nullable restore depois dela.
Você receberá um aviso de que RequestId não foi inicializado. Ao examinar a classe, você deve decidir que a
propriedade RequestId deve ser nula em alguns casos. A existência da propriedade ShowRequestId indica que é
possível faltar valores. Como null é válido, adicione o ? no tipo string para indicar que a propriedade
RequestId é um tipo de referência anulável. A classe final se parece com o seguinte exemplo:
#nullable enable
public class ErrorModel : PageModel
{
public string? RequestId { get; set; }
Verifique os usos da propriedade, e você verá que na página associada, a propriedade passa por uma verificação
de valor nulo antes da renderização na marcação. Esse é um uso seguro de um tipo de referência anulável,
portanto, você concluiu essa classe.
if (!string.IsNullOrEmpty(feedUrl))
{
try
{
NewsItems = await _newsService.GetNews(feedUrl);
}
catch (UriFormatException)
{
ErrorText = "There was a problem parsing the URL.";
return;
}
catch (WebException ex) when (ex.Status == WebExceptionStatus.NameResolutionFailure)
{
ErrorText = "Unknown host name.";
return;
}
catch (WebException ex) when (ex.Status == WebExceptionStatus.ProtocolError)
{
ErrorText = "Syndication feed not found.";
return;
}
catch (AggregateException ae)
{
ae.Handle((x) =>
{
if (x is XmlException)
{
ErrorText = "There was a problem parsing the feed. Are you sure that URL is a
syndication feed?";
return true;
}
return false;
});
}
}
}
}
Adicione a diretiva #nullable enable e você verá dois avisos. Nem a propriedade ErrorText , ou a propriedade
NewsItems é inicializada. Um exame dessa classe o levaria a acreditar que ambas as propriedades devem ser tipos
de referência anuláveis: ambas têm setters privados. Exatamente um é atribuído no método OnGet . Antes de fazer
alterações, examine os consumidores das duas propriedades. Na própria página, o ErrorText é verificado em
relação ao valor nulo antes de gerar a marcação dos erros. A coleção NewsItems é verificada com relação a null
e verificada para garantir que a coleção tenha itens. Uma correção rápida seria transformar as duas propriedades
em tipos de referência anuláveis. Uma correção melhor seria transformar a coleção em um tipo de referência não
anulável e adicionar itens à coleção existente ao recuperar notícias. A primeira correção é adicionar o ? ao tipo
string para o ErrorText :
Essa alteração não se propagará pelo outro código, pois qualquer acesso à propriedade ErrorText já foi
protegida por verificações de valor nulo. Em seguida, inicialize a lista NewsItems e remova a propriedade setter,
tornando-a uma propriedade somente leitura:
Isso corrigiu o aviso, mas introduziu um erro. Agora, a lista NewsItems foi corrigida pela construção, mas o
código que define a lista no OnGet deve ser alterado para coincidir com a nova API. Em vez de uma atribuição,
chame AddRange para adicionar os itens de notícias à lista existente:
NewsItems.AddRange(await _newsService.GetNews(feedUrl));
Usar AddRange em vez de uma atribuição significa que o método GetNews pode retornar um IEnumerable em vez
de um List . Isso salva uma alocação. Altere a assinatura do método e remova a chamada a ToList , conforme
mostra o exemplo de código a seguir:
// Something else
default:
break;
}
}
}
catch (AggregateException ae)
{
throw ae.Flatten();
}
}
A alteração da assinatura também interrompe testes de uma das opções. Abra o arquivo NewsServiceTests.cs na
pasta Services do projeto SimpleFeedReader.Tests . Navegue até o teste Returns_News_Stories_Given_Valid_Uri e
altere o tipo da variável result para IEnumerable<NewsItem> . A alteração do tipo significa que a propriedade
Count não está mais disponível. Portanto, substitua a propriedade Count no Assert com uma chamada para
Any() :
// Act
IEnumerable<NewsStoryViewModel> result =
await _newsService.GetNews(feedUrl);
// Assert
Assert.True(result.Any());
Você também precisará adicionar uma instrução using System.Linq ao início do arquivo.
Esse conjunto de alterações destaca uma consideração especial ao atualizar o código que inclui instanciações
genéricas. Tanto a lista quanto os elementos na lista de tipos não anuláveis. Um deles ou ambos podem ser tipos
anuláveis. Todas as declarações a seguir são permitidas:
List<NewsStoryViewModel> : lista não anulável de modelos de exibição não anuláveis.
List<NewsStoryViewModel?> : lista não anulável de modelos de exibição anuláveis.
List<NewsStoryViewModel>? : lista anulável de modelos de exibição não anuláveis.
List<NewsStoryViewModel?>? : lista anulável de modelos de exibição anuláveis.
O parâmetro IMapper é tipado como uma referência não anulável. Ele é chamado pelo código de infraestrutura
do ASP.NET Core, portanto, o compilador não sabe que o IMapper nunca será nulo. O contêiner de DI (injeção
de dependência) do ASP.NET Core padrão gera uma exceção se não puder resolver um serviço necessário, para
que o código fique correto. O compilador não consegue validar todas as chamadas para suas APIs públicas,
mesmo que seu código seja compilado com contextos de anotação anuláveis habilitados. Além disso, suas
bibliotecas podem ser consumidas por projetos que ainda não aceitaram o uso de tipos de referência anuláveis.
Valide as entradas para APIs públicas, mesmo que você as tenha declarado como tipos não anuláveis.
Obter o código
Você corrigiu os avisos identificados na compilação de teste inicial, portanto, agora você pode ativar o contexto de
anotação anulável para os dois projetos. Recompile os projetos; o compilador não relatará nenhum aviso. Você
pode obter o código do projeto concluído no repositório do GitHub dotnet/samples.
Os novos recursos que dão suporte aos tipos de referência anuláveis ajudam você a encontrar e corrigir possíveis
erros no modo de manipulação de valores null em seu código. A habilitação do contexto de anotação anulável
permite que você expresse sua intenção de design: algumas variáveis nunca devem ser nulas, outras variáveis
podem conter valores nulos. Esses recursos facilitam a declaração de sua intenção de design. Da mesma forma, o
contexto de aviso anulável instrui o compilador a emitir avisos quando você violar essa intenção. Esses avisos
servirão como orientação para criar atualizações que tornem seu código mais resiliente e menos propensa a
lançar uma NullReferenceException durante a execução. Você pode controlar o escopo desses contextos para se
concentrar na migração de áreas locais do código, enquanto a base de código restante permanece inalterada. Na
prática, você pode tornar essa tarefa de migração uma parte da manutenção regular das suas classes. Este tutorial
demonstrou o processo para migrar um aplicativo a fim de usar tipos de referência anuláveis. Você pode explorar
um exemplo real maior desse processo examinando a solicitação de pull feita por Jon Skeet para incorporar tipos
de referência anuláveis em NodaTime.
Tutorial: gerar e consumir fluxos assíncronos C#
usando o 8,0 e o .net Core 3,0
30/10/2019 • 14 minutes to read • Edit Online
O C#8.0 apresenta fluxos assíncronos, que modelam uma fonte de dados de streaming quando os elementos no
fluxo de dados podem ser recuperados ou gerados de forma assíncrona. Os fluxos assíncronos contam com novas
interfaces introduzidas no .NET Standard 2.1 e implementadas no .NET Core 3.0 para fornecer um modelo de
programação natural para fontes de dados de streaming assíncrono.
Neste tutorial, você aprenderá a:
Criar uma fonte de dados que gera uma sequência de elementos de dados de forma assíncrona.
Consumir essa fonte de dados de forma assíncrona.
Reconhecer quando a nova interface e a fonte de dados forem preferenciais para sequências anteriores de
dados síncronos.
Prerequisites
Você precisará configurar seu computador para executar o .NET Core, incluindo o C# compilador 8,0. O C#
compilador 8 está disponível a partir do Visual Studio 2019 versão 16,3 ou do SDK do .NET Core 3,0.
Você precisará criar um token de acesso do GitHub para poder acessar o ponto de extremidade GitHub GraphQL.
Selecione as seguintes permissões para o Token de acesso do GitHub:
repo:status
public_repo
Salve o token de acesso em um local seguro, de modo que possa usá-lo para acessar o ponto de extremidade da
API do GitHub.
WARNING
Mantenha seu token de acesso pessoal protegido. Qualquer software com seu token de acesso pessoal pode fazer chamadas
da API do GitHub usando seus direitos de acesso.
Este tutorial pressupõe que você esteja familiarizado com o C# e .NET, incluindo o Visual Studio ou a CLI do .NET
Core.
try
{
var results = await runPagedQueryAsync(client, PagedIssueQuery, "docs",
cancellationSource.Token, progressReporter);
foreach(var issue in results)
Console.WriteLine(issue);
}
catch (OperationCanceledException)
{
Console.WriteLine("Work has been cancelled");
}
}
Você pode definir uma variável de ambiente GitHubKey para o token de acesso pessoal ou pode substituir o último
argumento na chamada para GenEnvVariable com seu token de acesso pessoal. Não coloque o código de acesso
no código-fonte se for salvar a fonte com outras ou coloque-a em um repositório de origem compartilhado.
Após criar o cliente do GitHub, o código em Main criará um objeto de relatório de andamento e um token de
cancelamento. Depois que esses objetos forem criados, Main chamará runPagedQueryAsync para recuperar os 250
problemas mais recente criados. Depois que a tarefa for concluída, os resultados serão exibidos.
Ao executar o aplicativo inicial, você poderá realizar algumas observações importantes sobre como esse aplicativo
é executado. Você verá o progresso informado para cada página retornada do GitHub. É possível observar uma
pausa perceptível antes do GitHub retornar cada nova página de problemas. Por fim, os problemas só serão
exibidos depois que todas as 10 páginas forem recuperadas do GitHub.
Examinar a implementação
A implementação revela por que você observou o comportamento discutido na seção anterior. Examine o código
para runPagedQueryAsync :
private static async Task<JArray> runPagedQueryAsync(GitHubClient client, string queryText, string repoName,
CancellationToken cancel, IProgress<int> progress)
{
var issueAndPRQuery = new GraphQLRequest
{
Query = queryText
};
issueAndPRQuery.Variables["repo_name"] = repoName;
Vamos nos concentrar no algoritmo de paginação e na estrutura assíncrona do código anterior. (Você pode
consultar a documentação do GitHub GraphQL para obter detalhes sobre a API GraphQL do github.) O método
runPagedQueryAsync enumera os problemas do mais recente para o mais antigo. Ele solicita 25 problemas por
página e examina a estrutura pageInfo da resposta para continuar com a página anterior. Isso segue o suporte de
paginação padrão do GraphQL para respostas com várias páginas. A resposta inclui um objeto pageInfo que
inclui um valor hasPreviousPages e um valor startCursor usados para solicitar a página anterior. Os problemas
estão na matriz nodes . O método runPagedQueryAsync anexa esses nós em uma matriz que contém todos os
resultados de todas as páginas.
Após a recuperação e a restauração de uma página de resultados, runPagedQueryAsync informa o andamento e
verifica o cancelamento. Se o cancelamento tiver sido solicitado, runPagedQueryAsync gerará um
OperationCanceledException.
Há vários elementos nesse código que podem ser melhorados. Acima de tudo, runPagedQueryAsync deve alocar
armazenamento para todos os problemas retornados. Este exemplo é interrompido em 250 problemas porque
recuperar todos os problemas exigiria muito mais memória para armazenar todos os problemas recuperados.
Além disso, os protocolos que dão suporte ao progresso e ao cancelamento tornam o algoritmo mais difícil de
entender em sua primeira leitura. Você deve procurar a classe progresso para localizar onde o progresso é
informado. Você também tem que rastrear as comunicações por meio de CancellationTokenSource e seu
associado CancellationToken para entender onde o cancelamento foi solicitado e onde ele foi concedido.
namespace System.Collections.Generic
{
public interface IAsyncEnumerable<out T>
{
IAsyncEnumerator<T> GetAsyncEnumerator(CancellationToken cancellationToken = default);
}
ValueTask<bool> MoveNextAsync();
}
}
namespace System
{
public interface IAsyncDisposable
{
ValueTask DisposeAsync();
}
}
Essas três interfaces devem ser familiares à maioria dos desenvolvedores C#. Elas se comportam de maneira
semelhante às suas contrapartes síncronas:
System.Collections.Generic.IEnumerable<T>
System.Collections.Generic.IEnumerator<T>
System.IDisposable
Um tipo que pode não ser familiar é System.Threading.Tasks.ValueTask. A estrutura ValueTask fornece uma API
semelhante para a classe System.Threading.Tasks.Task. ValueTask é usado nas interfaces por motivos de
desempenho.
O código inicial processa cada página à medida que a página é recuperada, como mostrado no código a seguir:
finalResults.Merge(issues(results)["nodes"]);
progress?.Report(issuesReturned);
cancel.ThrowIfCancellationRequested();
Substitua essas três linhas pelo seguinte código:
Você também pode remover a declaração de finalResults anteriormente nesse método e a instrução return que
segue o loop que você modificou.
Você terminou as alterações para gerar um fluxo assíncrono. O método concluído deve se parecer com o código a
seguir:
Em seguida, você pode alterar o código que consome a coleção para consumir o fluxo assíncrono. Localize o
seguinte código em Main que processa a coleção de problemas:
var progressReporter = new progressStatus((num) =>
{
Console.WriteLine($"Received {num} issues in total");
});
CancellationTokenSource cancellationSource = new CancellationTokenSource();
try
{
var results = await runPagedQueryAsync(client, PagedIssueQuery, "docs",
cancellationSource.Token, progressReporter);
foreach(var issue in results)
Console.WriteLine(issue);
}
catch (OperationCanceledException)
{
Console.WriteLine("Work has been cancelled");
}
int num = 0;
await foreach (var issue in runPagedQueryAsync(client, PagedIssueQuery, "docs"))
{
Console.WriteLine(issue);
Console.WriteLine($"Received {++num} issues in total");
}
O C#7 introduziu recursos básicos de correspondência de padrões. Esses recursos foram estendidos no C# 8 com
novos padrões e expressões. É possível escrever uma funcionalidade que se comporte como se você tivesse
estendido tipos que poderiam estar em outras bibliotecas. Outro uso dos padrões é criar a funcionalidade de que
seu aplicativo precisa, mas que não é um recurso fundamental do tipo que está sendo estendido.
Neste tutorial, você aprenderá a:
Reconhecer situações em que a correspondência de padrões deverá ser usada.
Usar expressões de correspondência de padrões para implementar o comportamento com base em tipos e
valores de propriedade.
Combinar a correspondência de padrões com outras técnicas para criar algoritmos completos.
Prerequisites
Você precisará configurar seu computador para executar o .NET Core, incluindo o C# compilador 8,0. O C#
compilador 8 está disponível a partir do Visual Studio 2019 versão 16,3 ou do SDK do .NET Core 3,0.
Este tutorial pressupõe que você esteja familiarizado com o C# e .NET, incluindo o Visual Studio ou a CLI do .NET
Core.
namespace CommercialRegistration
{
public class DeliveryTruck
{
public int GrossWeightClass { get; set; }
}
}
namespace LiveryRegistration
{
public class Taxi
{
public int Fares { get; set; }
}
Faça o download do código inicial no repositório dotnet/samples do GitHub. É possível ver que as classes de
veículos são de sistemas diferentes, e estão em namespaces diferentes. Nenhuma base comum de classe, além da
System.Object pode ser aproveitada.
using System;
using CommercialRegistration;
using ConsumerVehicleRegistration;
using LiveryRegistration;
namespace toll_calculator
{
public class TollCalculator
{
public decimal CalculateToll(object vehicle) =>
vehicle switch
{
Car c => 2.00m,
Taxi t => 3.50m,
Bus b => 5.00m,
DeliveryTruck t => 10.00m,
{ } => throw new ArgumentException(message: "Not a known vehicle type", paramName:
nameof(vehicle)),
null => throw new ArgumentNullException(nameof(vehicle))
};
}
}
O código anterior usa uma expressão switch (não igual a uma instrução switch ) que testa o tipo de padrão. A
expressão switch inicia-se com a variável, vehicle no código anterior, seguida pela palavra-chave switch . Em
seguida, estão os braços switch dentro de chaves. A expressão switch faz outros refinamentos na sintaxe que
circunda a instrução switch . A palavra-chave case é omitida, e o resultado de cada braço é uma expressão. Os
dois últimos braços apresentam um novo recurso de linguagem. O caso { } corresponde a qualquer objeto não
nulo que não correspondia a um braço anterior. Este braço captura qualquer tipo incorreto passado para esse
método. O caso { } precisa seguir os casos para cada tipo de veículo. Se a ordem for revertida, o caso { } terá
precedência. Por fim, o padrão null detecta quando um null é passado para esse método. O padrão null
pode ser o último, porque os outros padrões de tipo correspondem apenas a um objeto não nulo do tipo correto.
Você pode testar esse código usando o seguinte código no Program.cs :
using System;
using CommercialRegistration;
using ConsumerVehicleRegistration;
using LiveryRegistration;
namespace toll_calculator
{
class Program
{
static void Main(string[] args)
{
var tollCalc = new TollCalculator();
try
{
tollCalc.CalculateToll("this will fail");
}
catch (ArgumentException e)
{
Console.WriteLine("Caught an argument exception when using the wrong type");
}
try
{
tollCalc.CalculateToll(null);
}
catch (ArgumentNullException e)
{
Console.WriteLine("Caught an argument exception when using null");
}
}
}
}
Esse código é incluído no projeto inicial, mas é comentado. Remova os comentários e você pode testar o que
escreveu.
Você está começando a ver como os padrões podem ajudar a criar algoritmos em que o código e os dados estão
separados. A expressão switch testa o tipo e produz valores diferentes com base nos resultados. Mas isso é
somente o começo.
vehicle switch
{
Car { Passengers: 0} => 2.00m + 0.50m,
Car { Passengers: 1 } => 2.0m,
Car { Passengers: 2} => 2.0m - 0.50m,
Car c => 2.00m - 1.0m,
// ...
};
Os três primeiros casos testam o tipo como um Car , em seguida, verificam o valor da propriedade Passengers .
Se ambos corresponderem, essa expressão é avaliada e retornada.
Você também expande os casos para táxis de maneira semelhante:
vehicle switch
{
// ...
// ...
};
vehicle switch
{
// ...
// ...
};
A autoridade de pedágio não está preocupada com o número de passageiros nos caminhões de carga. Em vez
disso, ela ajusta a quantidade de pedágios com base na classe de peso dos caminhões da seguinte maneira:
Os caminhões mais de 5000 quilos pagam uma taxa adicional de R$ 5,00.
Os caminhões leves abaixo de 3.000 lb recebem um desconto de US$ 2,00.
Essa regra é implementada com o código a seguir:
vehicle switch
{
// ...
O código anterior mostra a cláusula when de um braço switch. Você usa a cláusula when para testar as condições,
com exceção da igualdade, em uma propriedade. Ao terminar, o método será muito semelhante ao seguinte:
vehicle switch
{
Car { Passengers: 0} => 2.00m + 0.50m,
Car { Passengers: 1} => 2.0m,
Car { Passengers: 2} => 2.0m - 0.50m,
Car c => 2.00m - 1.0m,
{ } => throw new ArgumentException(message: "Not a known vehicle type", paramName: nameof(vehicle)),
null => throw new ArgumentNullException(nameof(vehicle))
};
Muitos desses braços switch são exemplos de padrões recursivos. Por exemplo, Car { Passengers: 1} mostra
um padrão constante dentro de um padrão de propriedade.
É possível fazer esse código menos repetitivo, usando switches aninhados. O Car e Taxi têm quatro braços
diferentes nos exemplos anteriores. Em ambos os casos, é possível criar um padrão de tipo que alimente um
padrão de propriedade. Essa técnica é mostrada no código a seguir:
public decimal CalculateToll(object vehicle) =>
vehicle switch
{
Car c => c.Passengers switch
{
0 => 2.00m + 0.5m,
1 => 2.0m,
2 => 2.0m - 0.5m,
_ => 2.00m - 1.0m
},
{ } => throw new ArgumentException(message: "Not a known vehicle type", paramName: nameof(vehicle)),
null => throw new ArgumentNullException(nameof(vehicle))
};
Na amostra anterior, o uso de uma expressão recursiva significa não repetir os braços Car e Taxi contendo
braços filho que testam o valor da propriedade. Essa técnica não é usada para os braços Bus e DeliveryTruck
porque esses estão testando intervalos da propriedade, e não valores discretos.
Há 16 combinações diferentes das três variáveis. Ao combinar algumas das condições, você simplificará a
expressão switch.
O sistema que coleta os pedágios usa uma estrutura DateTime para a hora em que o pedágio foi cobrado.
Construa métodos de membro que criam as variáveis da tabela anterior. A seguinte função usa como
correspondência de padrões a expressão switch para expressar se um DateTime representa um fim de semana ou
um dia útil:
Depois, adicione uma função semelhante para categorizar o tempo nos blocos:
O método anterior não usa a correspondência de padrões. Fica mais claro usando uma cascata familiar de
instruções if . Adicione um enum privado para converter cada intervalo de tempo em um valor discreto.
Depois de criar esses métodos, é possível usar outra expressão switch com o padrão de tupla para calcular o
preço premium. Você pode construir uma expressão switch com todos os 16 braços:
Tanto o tráfego de entrada quanto o de saída têm o mesmo multiplicador durante o dia e a noite, nos dias úteis.
Esses quatro braços switch podem ser substituídos por estas duas linhas:
Por fim, você pode remover os dois horários de pico em que é pago o preço normal. Quando remover essas
braços, substitua o false por um descarte ( _ ) no braço switch final. O método concluído será o seguinte:
Este exemplo destaca uma das vantagens da correspondência de padrões: os branches de padrões são avaliados
na ordem. Se você os reorganizar para que um branch anterior trate um dos casos posteriores, o compilador
emitirá um aviso sobre o código inacessível. Essas regras de linguagem tornam as simplificações anteriores mais
fáceis com a certeza de que o código não foi alterado.
A correspondência de padrões torna alguns tipos de código mais legíveis e oferece uma alternativa às técnicas
orientadas a objeto quando não é possível adicionar o código às classes. A nuvem está fazendo com que os dados
e a funcionalidade existam separadamente. A forma dos dados e as operações nela não são necessariamente
descritas juntas. Neste tutorial, você utilizou os dados existentes de maneiras completamente diferentes de sua
função original. A correspondência de padrões proporcionou a capacidade de escrever a funcionalidade que
substituiu esses tipos, ainda que não tenha sido possível estendê-los.
Próximas etapas
Baixe o código concluído no repositório dotnet/samples do GitHub. Explore os padrões por conta própria e
adicione essa técnica em suas atividades regulares de codificação. Aprender essas técnicas lhe oferece outra
maneira de abordar problemas e criar novas funcionalidades.
Aplicativo do Console
30/10/2019 • 22 minutes to read • Edit Online
Este tutorial ensina vários recursos no .NET Core e da linguagem C#. Você aprenderá:
As noções básicas da CLI (Interface de Linha de Comando) do .NET Core
A estrutura de um aplicativo de console C#
E/S do Console
Fundamentos das APIs de E/S de arquivo no .NET
Os fundamentos da programação assíncrona controlada por tarefas no .NET Core
Você compilará um aplicativo que lê um arquivo de texto e exibe o conteúdo desse arquivo de texto no console. A
saída para o console é conduzida a fim de corresponder à leitura em voz alta. É possível acelerar ou diminuir o
ritmo pressionando as teclas "<" (menor que) ou ">" (maior que).
Há vários recursos neste tutorial. Vamos compilá-los individualmente.
Prerequisites
Você precisará configurar seu computador para executar o .NET Core. Você pode encontrar as instruções de
instalação na página de downloads do .NET Core . Execute esse aplicativo no Windows, Linux, macOS ou em um
contêiner do Docker. Você precisará instalar o editor de código de sua preferência.
Criar o aplicativo
A primeira etapa é criar um novo aplicativo. Abra um prompt de comando e crie um novo diretório para seu
aplicativo. Torne ele o diretório atual. Digite o comando dotnet new console no prompt de comando. Isso cria os
arquivos iniciais de um aplicativo "Olá, Mundo" básico.
Antes de começar as modificações, vamos percorrer as etapas para execução do aplicativo simples Hello World.
Depois de criar o aplicativo, digite dotnet restore no prompt de comando. Esse comando executa o processo de
restauração do pacote NuGet. O NuGet é um gerenciador de pacotes do .NET. Esse comando baixa qualquer uma
das dependências ausentes para seu projeto. Como esse é um novo projeto, nenhuma das dependências foram
aplicadas, portanto, a primeira execução baixará a estrutura do .NET Core. Após essa etapa inicial, você só
precisará executar o dotnet restore ao adicionar novos pacotes dependentes, ou atualizar as versões de qualquer
uma de suas dependências.
NOTE
Começando com o SDK do .NET Core 2.0, não é necessário executar dotnet restore , pois ele é executado implicitamente
por todos os comandos que exigem uma restauração, como dotnet new , dotnet build e dotnet run . Ainda é um
comando válido em determinados cenários em que realizar uma restauração explícita faz sentido, como builds de integração
contínua no Azure DevOps Services ou em sistemas de build que precisam controlar explicitamente o horário em que a
restauração ocorrerá.
Depois de restaurar os pacotes, execute dotnet build . Isso executa o mecanismo de compilação e cria o
executável de seu aplicativo. Por fim, execute dotnet run para executar o aplicativo.
O código simples do aplicativo Hello World está totalmente em Program.cs. Abra esse arquivo com o seu editor
de texto favorito. Estamos prestes a fazer nossas primeiras alterações. Na parte superior do arquivo, confira uma
instrução using:
using System;
Essa instrução informa ao compilador que quaisquer tipos do namespace System estão dentro do escopo. Assim
como em outras linguagens orientadas a objeto que você pode ter usado, o C# usa namespaces para organizar
tipos. Este programa Olá, Mundo não é diferente. Você pode ver que o programa é encerrado no namespace com
o nome baseado no nome do diretório atual. Para este tutorial, vamos alterar o nome do namespace para
TeleprompterConsole :
namespace TeleprompterConsole
Esse método usa tipos de dois namespaces novos. Para que isso seja compilado, será necessário adicionar as duas
linhas a seguir na parte superior do arquivo:
using System.Collections.Generic;
using System.IO;
Execute o programa (usando dotnet run , e você poderá ver todas as linhas impressa no console).
Em seguida, será necessário modificar a forma como você consume as linhas do arquivo, e adicionar um atraso
depois de escrever cada palavra. Substitua a instrução Console.WriteLine(line) no método Main pelo seguinte
bloco:
Console.Write(line);
if (!string.IsNullOrWhiteSpace(line))
{
var pause = Task.Delay(200);
// Synchronously waiting on a task is an
// anti-pattern. This will get fixed in later
// steps.
pause.Wait();
}
A classe Task é o namespace System.Threading.Tasks, portanto, você precisa adicionar essa instrução using na
parte superior do arquivo:
using System.Threading.Tasks;
Execute o exemplo e verifique a saída. Agora, cada palavra única é impressa, seguida por um atraso de 200 ms. No
entanto, a saída exibida mostra alguns problemas, pois o arquivo de texto de origem contém várias linhas com
mais de 80 caracteres sem uma quebra de linha. Isso pode ser difícil de ler durante a rolagem da tela. Mas
também é fácil de corrigir. Apenas mantenha o controle do comprimento de cada linha e gere uma nova linha
sempre que o comprimento atingir um certo limite. Declare uma variável local após a declaração de words no
método ReadFrom que contém o comprimento da linha:
var lineLength = 0;
Em seguida, adicione o seguinte código após a instrução yield return word + " "; (antes da chave de
fechamento):
lineLength += word.Length + 1;
if (lineLength > 70)
{
yield return Environment.NewLine;
lineLength = 0;
}
Execute o exemplo e você poderá ler em voz alta de acordo com o ritmo pré-configurado.
Tarefas assíncronas
Nesta etapa final, você adicionará o código para gravar a saída de forma assíncrona em uma tarefa, enquanto
executa também outra tarefa para ler a entrada do usuário, casos ele queira acelerar ou diminuir o ritmo da
exibição do texto ou interromper a exibição do texto por completo. Essa etapa tem alguns passos e, no final, você
terá todas as atualizações necessárias. A primeira etapa é criar um método de retorno Task assíncrono que
representa o código que você criou até agora para ler e exibir o arquivo.
Adicione este método à sua classe Program (ele é obtido do corpo de seu método Main ):
Você observará duas alterações. Primeiro, no corpo do método, em vez de chamar Wait() para aguardar de forma
síncrona a conclusão de uma tarefa, essa versão usa a palavra-chave await . Para fazer isso, você precisa adicionar
o modificador async à assinatura do método. Esse método retorna Task . Observe que não há instruções return
que retornam um objeto Task . Em vez disso, esse objeto Task é criado pelo código gerado pelo compilador
quando você usa o operador await . Você pode imaginar que esse método retorna quando atinge um await . A
Task retornada indica que o trabalho não foi concluído. O método será retomado quando a tarefa em espera for
concluída. Após a execução completa, a Task retornada indicará a conclusão. O código de chamada pode
monitorar essa Task retornada para determinar quando ela foi concluída.
Chame esse novo método em seu método Main :
ShowTeleprompter().Wait();
Aqui, em Main , o código aguarda de forma síncrona. Use o operador await em vez de esperar de forma síncrona
sempre que possível. Porém, no método Main do aplicativo de console, não é possível usar o operador await .
Isso resultaria no encerramento do aplicativo antes da conclusão de todas as tarefas.
NOTE
Caso use o C# 7.1 ou posterior, você poderá criar aplicativos de console com o método async Main .
Em seguida, é necessário escrever o segundo método assíncrono a ser lido no Console e ficar atento às teclas "<"
(menor que), ">" (maior que) e "X" ou "x". Este é o método que você adiciona à tarefa:
Isso cria uma expressão lambda para representar um delegado Action que lê uma chave no Console e modifica
uma variável local que representa o atraso quando o usuário pressiona as teclas "<" (menor que) ou ">" (maior
que). O método de delegado termina quando o usuário pressiona a tecla "X" ou "x", que permitem ao usuário
interromper a exibição de texto a qualquer momento. Esse método usa ReadKey() para bloquear e aguardar até
que o usuário pressione uma tecla.
Para concluir esse recurso, você precisa criar um novo método de retorno async Task que inicia essas duas
tarefas ( GetInput e ShowTeleprompter ) e também gerencia os dados compartilhados entre essas tarefas.
É hora de criar uma classe que pode manipular os dados compartilhados entre essas duas tarefas. Essa classe
contém duas propriedades públicas: o atraso e um sinalizador Done para indicar que o arquivo foi lido
completamente:
namespace TeleprompterConsole
{
internal class TelePrompterConfig
{
public int DelayInMilliseconds { get; private set; } = 200;
Coloque essa classe em um novo arquivo e cerque-a pelo namespace TeleprompterConsole , conforme mostrado
acima. Também é necessário adicionar uma instrução using static para que você possa fazer referência ao
método Min e Max sem os nomes de classe ou namespace delimitadores. Uma instrução using static importa
os métodos de uma classe. Isso é o oposto das instruções using usadas até este ponto, as quais importaram
todas as classes de um namespace.
Em seguida, atualize os métodos ShowTeleprompter e GetInput para usar o novo objeto config . Escreva um
método final async de retorno de Task para iniciar as duas tarefas e sair quando a primeira tarefa for concluída:
O novo método aqui é a chamada WhenAny(Task[]). Isso cria uma Task que termina assim que qualquer uma
das tarefas na lista de argumentos for concluída.
Depois, atualize os métodos ShowTeleprompter e GetInput para usar o objeto config para o atraso:
private static async Task ShowTeleprompter(TelePrompterConfig config)
{
var words = ReadFrom("sampleQuotes.txt");
foreach (var word in words)
{
Console.Write(word);
if (!string.IsNullOrWhiteSpace(word))
{
await Task.Delay(config.DelayInMilliseconds);
}
}
config.SetDone();
}
Essa nova versão de ShowTeleprompter chama um novo método na classe TeleprompterConfig . Agora, você
precisa atualizar Main para chamar RunTeleprompter em vez de ShowTeleprompter :
RunTeleprompter().Wait();
Conclusão
Este tutorial mostrou a você alguns recursos da linguagem C# e as bibliotecas .NET Core relacionadas ao trabalho
em aplicativos de Console. Use esse conhecimento como base para explorar mais sobre a linguagem e sobre as
classes apresentadas aqui. Você já viu os fundamentos de E/S do Arquivo e do Console, uso com bloqueio e sem
bloqueio da programação assíncrona controlada por tarefa, um tour pela linguagem C# e como os programas em
C# são organizados, além da Interface de linha de comando e ferramentas do .NET Core.
Para obter mais informações sobre E/S de arquivo, consulte o tópico E/S de arquivo e de fluxo. Para obter mais
informações sobre o modelo de programação assíncrona usado neste tutorial, consulte os tópicos Programação
assíncrona controlada por tarefas e Programação assíncrona.
Cliente REST
23/10/2019 • 28 minutes to read • Edit Online
Introdução
Este tutorial ensina vários recursos no .NET Core e da linguagem C#. Você aprenderá:
As noções básicas da CLI (Interface de Linha de Comando) do .NET Core.
Uma visão geral dos recursos da linguagem C#.
Gerenciamento de dependências com o NuGet
Comunicações HTTP
Processamento de informações de JSON
Gerenciamento de configuração com Atributos.
Você compilará um aplicativo que emite solicitações HTTP para um serviço REST no GitHub. Você lerá
informações no formato JSON e converterá esse pacote JSON em objetos C#. Por fim, você verá como trabalhar
com objetos C#.
Há vários recursos neste tutorial. Vamos compilá-los individualmente.
Se preferir acompanhar com o exemplo final para esse tópico, você poderá baixá-lo. Para obter instruções de
download, consulte Exemplos e tutoriais.
Pré-requisitos
Você precisará configurar seu computador para executar o .NET Core. Você pode encontrar as instruções de
instalação na página de downloads do .NET Core . Execute esse aplicativo no Windows, Linux, macOS ou em um
contêiner do Docker. Será necessário instalar o editor de código de sua preferência. As descrições a seguir usam o
Visual Studio Code, que é uma software livre, no editor de plataforma. No entanto, você pode usar quaisquer
ferramentas que esteja familiarizado.
Criar o aplicativo
A primeira etapa é criar um novo aplicativo. Abra um prompt de comando e crie um novo diretório para seu
aplicativo. Torne ele o diretório atual. Digite o comando dotnet new console no prompt de comando. Isso cria os
arquivos iniciais de um aplicativo "Olá, Mundo" básico. Como esse é um novo projeto, nenhuma das dependências
está em vigor, portanto, a primeira execução baixará o .NET Core Framework, instalará um certificado de
desenvolvimento e executará o gerenciador de pacotes do NuGet para restaurar dependências ausentes.
Antes de começar a fazer modificações, digite dotnet run (veja a observação) no prompt de comando para
executar seu aplicativo. dotnet run executará automaticamente dotnet restore se estiverem faltando
dependências em seu ambiente. Ele também executará dotnet build se seu aplicativo precisar ser reconstruído.
Após sua configuração inicial, você só precisará executar dotnet restore ou dotnet build quando fizer sentido
para seu projeto.
<Project Sdk="Microsoft.NET.Sdk">
<ItemGroup>
<PackageReference Include="System.Runtime.Serialization.Json" Version="4.3.0" />
</ItemGroup>
A maioria dos editores de código fornece conclusão para versões diferentes dessas bibliotecas. Convém usar a
versão mais recente de qualquer pacote que você adicionar. No entanto, é importante ter certeza de que as
versões de todos os pacotes correspondam, e que também correspondam à versão da estrutura de aplicativo do
.NET Core.
Após fazer essas alterações, execute dotnet restore (veja a observação) para que o pacote seja instalado em seu
sistema.
Será necessário adicionar uma instrução using na parte superior de seu método Main para que o compilador de
C# reconheça o tipo Task:
using System.Threading.Tasks;
Se você compilar o projeto neste momento, receberá um aviso gerado para esse método, pois ele não contém
operadores await e executará de forma síncrona. Ignore isso por enquanto. Adicione os operadores await à
medida que você preenche o método.
Em seguida, renomeie o namespace definido na instrução namespace , alterando o padrão de ConsoleApp para
WebAPIClient . Posteriormente, definiremos uma classe repo neste namespace.
Em seguida, atualize o método Main para chamar esse método. O método ProcessRepositories retorna uma
Tarefa, e você não deve sair do programa antes da conclusão dessa tarefa. Portanto, use o método Wait para
bloquear e esperar a conclusão da tarefa:
static void Main(string[] args)
{
ProcessRepositories().Wait();
}
Agora, você tem um programa que não faz nada, mas o faz de forma assíncrona. Vamos melhorar isso.
Primeiro você precisa de um objeto que é capaz de recuperar dados da Web; você pode usar um HttpClient para
fazer isso. Esse objeto manipula a solicitação e as respostas. Crie uma única instância desse tipo na classe Program
dentro do arquivo Program.cs.
namespace WebAPIClient
{
class Program
{
private static readonly HttpClient client = new HttpClient();
Também será necessário adicionar duas instruções using novas na parte superior do arquivo para que isso seja
compilado:
using System.Net.Http;
using System.Net.Http.Headers;
A primeira versão faz uma solicitação da Web para ler a lista de todos os repositórios na organização dotnet
foundation. (A ID do gitHub para o .NET Foundation é 'dotnet'). As primeiras linhas configuram o HttpClient para
essa solicitação. Primeiro, ele é configurado para aceitar as respostas JSON do GitHub. Esse formato é
simplesmente JSON. A próxima linha adiciona um cabeçalho de agente do usuário para todas as solicitações deste
objeto. Esses dois cabeçalhos são verificados pelo código do servidor GitHub e são necessários para recuperar
informações do GitHub.
Depois de configurar o HttpClient, faça uma solicitação da Web e recupere a resposta. Nesta primeira versão, você
usa o método de conveniência HttpClient.GetStringAsync(String). Este método prático inicia uma tarefa que faz a
solicitação da Web e, depois, quando a solicitação retornar, ele lê o fluxo de resposta e extrai o conteúdo do fluxo.
O corpo da resposta retorna como um String. A cadeia de caracteres fica disponível após a conclusão da tarefa.
As duas últimas linhas desse método aguardam a tarefa e, depois, imprimem a resposta no console. Compilar o
aplicativo e executá-lo. O aviso de compilação desapareceu, pois agora o ProcessRepositories contêm um
operador await . Você verá uma longa exibição do texto formatado em JSON.
using System;
namespace WebAPIClient
{
public class repo
{
public string name;
}
}
Coloque o código acima em um novo arquivo chamado 'repo.cs'. Esta versão da classe representa o caminho mais
simples para processar os dados em JSON. O nome da classe e o nome do membro corresponderem aos nomes
usados no pacote JSON, em vez de seguir as convenções em C#. Corrija isso mais tarde fornecendo alguns
atributos de configuração. Essa classe demonstra outro recurso importante da serialização e desserialização
JSON: Nem todos os campos no pacote JSON fazem parte dessa classe. O serializador JSON ignorará as
informações que não estão incluídas no tipo de classe que está sendo usado. Esse recurso facilita a criação de tipos
que funcionam apenas com um subconjunto dos campos no pacote JSON.
Agora que você criou o tipo, vamos desserializá-lo. Você precisará criar um objeto DataContractJsonSerializer.
Este objeto deve saber o tipo de CLR esperada para o pacote JSON que ele recupera. O pacote do GitHub contém
uma sequência de repositórios, então um List<repo> é o tipo correto. Adicione a seguinte linha ao seu método
ProcessRepositories :
Você está usando dois novos namespaces, portanto será necessário adicioná-los também:
using System.Collections.Generic;
using System.Runtime.Serialization.Json;
Em seguida, você usará o serializador para converter JSON em objetos de C#. Substitua a chamada para
GetStringAsync(String) em seu método ProcessRepositories pelas duas linhas a seguir:
Observe que agora você está usando GetStreamAsync(String) em vez de GetStringAsync(String). O serializador
usa um fluxo, em vez de uma cadeia de caracteres, como sua fonte. Vamos explicar alguns recursos da linguagem
C# que estão sendo usados na segunda linha acima. O argumento ReadObject(Stream) é uma expressão await .
Expressões await podem aparecer em quase todo lugar em seu código, apesar de que até o momento, você apenas
as viu como parte de uma instrução de atribuição.
Em segundo lugar, o operador as converte do tipo de tempo de compilação do object para List<repo> . A
declaração de ReadObject(Stream) declara que ele retorna um objeto do tipo System.Object. ReadObject(Stream)
retornará o tipo especificado no momento da construção ( List<repo> neste tutorial). Se a conversão não tiver
êxito, o operador as será avaliado com null , em vez de gerar uma exceção.
Você está quase terminando esta seção. Agora que você converteu o JSON em objetos C#, vamos exibir o nome
de cada repositório. Substitua as linhas que mostram:
pelo seguinte:
Compile e execute o aplicativo. Ele imprimirá os nomes dos repositórios que fazem parte do .NET Foundation.
Controle da serialização
Antes de adicionar mais recursos, vamos abordar o tipo repo e fazê-lo seguir convenções mais padrão de C#.
Faça isso anotando a tipo repo com atributos que controlem o modo como o serializador JSON funciona. Em seu
caso, você usará esses atributos para definir um mapeamento entre os nomes de chave JSON e os nomes de
classe e de membros de C#. Os dois atributos usados são os atributos DataContractAttribute e
DataMemberAttribute. Por convenção, todas as classes de atributo terminam com o sufixo Attribute . No entanto,
não é necessário usar esse sufixo ao aplicar um atributo.
Os atributos DataContractAttribute e DataMemberAttribute são atributos em uma biblioteca diferente, então você
precisará adicionar essa biblioteca ao seu arquivo de projeto C# como uma dependência. Adicione a seguinte linha
à seção <ItemGroup> de seu arquivo de projeto:
Depois de salvar o arquivo, execute dotnet restore (veja a observação) para recuperar esse pacote.
Em seguida, abra o arquivo repo.cs . Vamos alterar o nome para usar Pascal Case e soletrar o nome Repository
completo. Ainda queremos mapear os nós de 'repositório' de JSON para esse tipo, então será necessário adicionar
o atributo DataContractAttribute à declaração de classe. Você definirá a propriedade Name do atributo como o
nome de nós de JSON mapeados para esse tipo:
[DataContract(Name="repo")]
public class Repository
using System.Runtime.Serialization;
Você alterou o nome da classe repo para Repository , portanto, será necessário fazer a mesma alteração em
Program.cs (alguns editores podem oferecer suporte a uma refatoração de renomeação que fará essa alteração
automaticamente:)
var serializer = new DataContractJsonSerializer(typeof(List<Repository>));
// ...
Em seguida, vamos fazer a mesma alteração com o campo name usando a classe DataMemberAttribute. Faça as
seguintes alterações na declaração do campo name em repo.cs:
[DataMember(Name="name")]
public string Name;
Essa alteração significa que você precisa alterar o código que grava o nome de cada repositório em program.cs:
Console.WriteLine(repo.Name);
Faça uma dotnet build seguido por um dotnet run para certificar-se de que você tem os mapeamentos corretos.
Você deve ver o mesmo resultado de antes. Antes, processamos mais propriedades do servidor Web, vamos fazer
mais uma alteração na classe Repository . O membro Name é um campo acessível publicamente. Essa não é uma
boa prática orientada a objeto, então vamos alterá-lo para uma propriedade. Para nossos propósitos, não é
necessário executar nenhum código específico ao obter ou configurar a propriedade, mas a alteração de uma
propriedade facilita a adição dessas alterações posteriormente sem interromper qualquer código que usa a classe
Repository .
O compilador gera o corpo dos acessadores get e set , bem como um campo particular para armazenar o
nome. Seria semelhante ao código a seguir, que você pode digitar manualmente:
Vamos fazer mais uma alteração antes de adicionar novos recursos. O método ProcessRepositories pode fazer o
trabalho assíncrono e retornar uma coleção de repositórios. Vamos retornar o List<Repository> desse método e
mover o código que grava as informações no método Main .
Altere a assinatura de ProcessRepositories para retornar uma tarefa cujo resultado é uma lista de objetos
Repository :
O acesso à propriedade Result de uma Tarefa é bloqueado até que a tarefa seja concluída. Normalmente, seria
preferível await (aguardar) a conclusão da tarefa, como no método ProcessRepositories , mas isso não é
permitido no método Main .
[DataMember(Name="description")]
public string Description { get; set; }
[DataMember(Name="html_url")]
public Uri GitHubHomeUrl { get; set; }
[DataMember(Name="homepage")]
public Uri Homepage { get; set; }
[DataMember(Name="watchers")]
public int Watchers { get; set; }
Essas propriedades têm conversões internas do tipo cadeia de caracteres (que é o que os pacotes JSON contêm)
para o tipo de destino. O tipo Uri pode ser novidade para você. Ele representa um URI, ou, nesse caso, uma URL.
No caso dos tipos Uri e int , se o pacote JSON contiver dados que não são convertidos para o tipo de destino, a
ação de serialização lançará uma exceção.
Depois de adicioná-los, atualize o método Main para exibir esses elementos:
Como etapa final, vamos adicionar as informações para a última operação de envio por push. Essas informações
são formatadas dessa maneira na resposta JSON:
2016-02-08T21:27:00Z
Esse formato não segue o formato DateTime padrão do .NET. Por isso, você precisará escrever um método de
conversão personalizado. Provavelmente você também não quer expor a cadeia de caracteres bruta aos usuários
da classe Repository . Os atributos podem ajudar a controlar isto também. Primeiro, defina uma propriedade
private que conterá a representação de cadeia de caracteres da data e hora em sua classe Repository :
[DataMember(Name="pushed_at")]
private string JsonDate { get; set; }
O atributo DataMemberAttribute informa ao serializador de que isso deve ser processado, mesmo que não seja
um membro público. Em seguida, você precisa escrever uma propriedade pública somente leitura que converte a
cadeia de caracteres em um objeto DateTime válido, e retorna esse DateTime:
[IgnoreDataMember]
public DateTime LastPush
{
get
{
return DateTime.ParseExact(JsonDate, "yyyy-MM-ddTHH:mm:ssZ", CultureInfo.InvariantCulture);
}
}
Vamos falar sobre as novas construções acima. O atributo IgnoreDataMember instrui o serializador que esse tipo
não deve ser lido para ou gravado de qualquer objeto JSON. Essa propriedade contém apenas um acessador get
. Não há nenhum acessador set . É assim que você define uma propriedade somente leitura em C#. (Sim, você
pode criar propriedades somente gravação em C#, mas o valor delas é limitado.) O método ParseExact(String,
String, IFormatProvider) analisa uma cadeia de caracteres e cria um objeto DateTime usando um formato de data
fornecido e adiciona outros metadados a DateTime usando um objeto CultureInfo . Se a operação de análise
falhar, o acessador da propriedade gerará uma exceção.
Para usar InvariantCulture, você precisará adicionar o namespace System.Globalization às instruções using em
repo.cs :
using System.Globalization;
Por fim, adicione mais uma instrução de saída no console, e você estará pronto para compilar e executar esse
aplicativo novamente:
Console.WriteLine(repo.LastPush);
Conclusão
Este tutorial mostrou como fazer solicitações da Web, analisar o resultados e exibir as propriedades dos
resultados. Você também adicionou novos pacotes como dependências em seu projeto. Você viu alguns dos
recursos da linguagem C# que dão suporte a técnicas orientadas a objeto.
NOTE
Começando com o SDK do .NET Core 2.0, não é necessário executar dotnet restore , pois ele é executado implicitamente
por todos os comandos que exigem uma restauração, como dotnet new , dotnet build e dotnet run . Ainda é um
comando válido em determinados cenários em que realizar uma restauração explícita faz sentido, como builds de integração
contínua no Azure DevOps Services ou em sistemas de build que precisam controlar explicitamente o horário em que a
restauração ocorrerá.
Herança em C# e .NET
31/10/2019 • 44 minutes to read • Edit Online
Este tutorial apresenta a herança em C#. Herança é um recurso das linguagens de programação orientadas a
objeto que permite a definição de uma classe base que, por sua vez, fornece uma funcionalidade específica (dados
e comportamento), e a definição de classes derivadas que herdam ou substituem essa funcionalidade.
Prerequisites
Este tutorial pressupõe que você instalou o SDK do .NET Core. Visite a página de downloads do .NET Core para
baixá-lo. Você também precisa de um editor de código. Este tutorial usa o Visual Studio Code, embora você possa
usar qualquer editor de código que quiser.
NOTE
Começando com o SDK do .NET Core 2.0, não é necessário executar dotnet restore , pois ele é executado implicitamente
por todos os comandos que exigem uma restauração, como dotnet new , dotnet build e dotnet run . Ainda é um
comando válido em determinados cenários em que realizar uma restauração explícita faz sentido, como builds de integração
contínua no Azure DevOps Services ou em sistemas de build que precisam controlar explicitamente o horário em que a
restauração ocorrerá.
using System;
public class A
{
private int value = 10;
public class B : A
{
public int GetValue()
{
return this.value;
}
}
}
public class C : A
{
// public int GetValue()
// {
// return this.value;
// }
}
public class B : A
{ }
Classes derivadas também podem substituir membros herdados fornecendo uma implementação alternativa. Para
poder substituir um membro, o membro na classe base deve ser marcado com a palavra-chave virtual. Por padrão,
os membros da classe base não são marcados como virtual e não podem ser substituídos. A tentativa de
substituir um membro não virtual, como faz o exemplo a seguir, gera o erro do compilador CS0506: "O
<member> não pode substituir o membro herdado <member>, pois não está marcado como virtual, abstrato ou
de substituição".
public class A
{
public void Method1()
{
// Do something.
}
}
public class B : A
{
public override void Method1() // Generates CS0506.
{
// Do something else.
}
}
Em alguns casos, uma classe derivada deve substituir a implementação da classe base. Membros de classe base
marcados com a palavra-chave abstrato exigem que as classes derivadas os substituam. A tentativa de compilar o
exemplo a seguir gera um erro do compilador CS0534, a "<classe> não implementa o membro abstrato herdado
<membro>", pois a classe B não fornece uma implementação para A.Method1 .
public abstract class A
{
public abstract void Method1();
}
A herança se aplica apenas a classes e interfaces. Outras categorias de tipo (structs, delegados e enumerações) não
dão suporte à herança. Devido a essas regras, a tentativa de compilar o código como no exemplo a seguir produz
o erro do compilador CS0527: "O tipo 'ValueType' na lista de interfaces não é uma interface". A mensagem de erro
indica que, embora você possa definir as interfaces implementadas por um struct, não há suporte para a herança.
using System;
Herança implícita
Apesar dos tipos possíveis de herança por meio de herança única, todos os tipos no sistema de tipo .NET herdam
implicitamente de Object ou de um tipo derivado dele. A funcionalidade comum de Object está disponível para
qualquer tipo.
Para ver o que significa herança implícita, vamos definir uma nova classe, SimpleClass , que é simplesmente uma
definição de classe vazia:
É possível usar reflexão (o que permite inspecionar os metadados de um tipo para obter informações sobre esse
tipo) para obter uma lista dos membros que pertencem ao tipo SimpleClass . Embora você ainda não tenha
definido membros na classe SimpleClass , a saída do exemplo indica que, na verdade, ela tem nove membros. Um
desses membros é um construtor sem parâmetros (ou padrão) que é fornecido automaticamente para o tipo
SimpleClass pelo compilador de C#. Os oito são membros do Object, o tipo do qual todas as classes e interfaces
no sistema do tipo .NET herdam implicitamente.
using System;
using System.Reflection;
}
}
}
// The example displays the following output:
// Type SimpleClass has 9 members:
// ToString (Method): Public, Declared by System.Object
// Equals (Method): Public, Declared by System.Object
// Equals (Method): Public Static, Declared by System.Object
// ReferenceEquals (Method): Public Static, Declared by System.Object
// GetHashCode (Method): Public, Declared by System.Object
// GetType (Method): Public, Declared by System.Object
// Finalize (Method): Internal, Declared by System.Object
// MemberwiseClone (Method): Internal, Declared by System.Object
// .ctor (Constructor): Public, Declared by SimpleClass
A herança implícita da classe Object torna esses métodos disponíveis para a classe SimpleClass :
O método ToString público, que converte um objeto SimpleClass em sua representação de cadeia de
caracteres, retorna o nome de tipo totalmente qualificado. Nesse caso, o método ToString retorna a cadeia
de caracteres "SimpleClass".
Três métodos de teste de igualdade de dois objetos: o método Equals(Object) da instância pública, o
método Equals(Object, Object) do público estático e o método ReferenceEquals(Object, Object) de
público estático. Por padrão, esses métodos testam a igualdade de referência; ou seja, para que seja iguais,
duas variáveis de objeto devem fazer referência ao mesmo objeto.
O método público GetHashCode , que calcula um valor que permite o uso de uma instância do tipo em
conjuntos de hash.
O método público GetType , que retorna um objeto Type que representa o tipo SimpleClass .
O método protegido Finalize, que é projetado para liberar recursos não gerenciados antes que a memória
de um objeto seja reivindicada pelo coletor de lixo.
O método protegido MemberwiseClone, que cria um clone superficial do objeto atual.
Devido à herança implícita, você pode chamar qualquer membro herdado de um objeto SimpleClass como se ele
fosse, na verdade, um membro definido na classe SimpleClass . Por exemplo, o exemplo a seguir chama o método
SimpleClass.ToString , que SimpleClass herda de Object.
using System;
A tabela a seguir lista as categorias de tipos que você pode criar em C#, e os tipos de onde eles herdam
implicitamente. Cada tipo base disponibiliza um conjunto diferente de membros por meio de herança para tipos
derivados implicitamente.
classe Object
NOTE
Uma classe ou struct pode implementar uma ou mais interfaces. Embora a implementação da interface seja apresentada
geralmente como uma alternativa para herança única, ou como uma forma de usar a herança com structs, ela tem como
objetivo expressar um relacionamento diferente (um relacionamento "pode fazer") entre uma interface e seu tipo de
implementação em comparação com a herança. Uma interface define um subconjunto de funcionalidades (como a
capacidade de testar a igualdade, comparar ou classificar objetos ou dar suporte à formatação e análise sensível à cultura)
que disponibiliza para seus tipos de implementação.
Observe que "é um(a)" também expressa o relacionamento entre um tipo e uma instanciação específica desse tipo.
No exemplo a seguir, Automobile é uma classe que tem três propriedades somente leitura exclusivas: Make , o
fabricante do automóvel; Model , o tipo de automóvel; e Year , o ano de fabricação. A classe Automobile também
tem um construtor cujos argumentos são atribuídos aos valores de propriedade. Ela também substitui o método
Object.ToString para produzir uma cadeia de caracteres que identifica exclusivamente a instância Automobile em
vez da classe Automobile .
using System;
if (model == null)
throw new ArgumentNullException("The model cannot be null.");
else if (String.IsNullOrWhiteSpace(model))
throw new ArgumentException("model cannot be an empty string or have space characters only.");
Model = model;
Nesse caso, você não deve depender da herança para representar marcas e modelos de carro específicos. Por
exemplo, você não precisa definir um tipo Packard para representar automóveis fabricados pela empresa Packard
Motor Car. Nesse caso, é possível representá-los criando um objeto Automobile com os valores apropriados
passados ao construtor de classe, como no exemplo a seguir.
using System;
Um relacionamento é-um(a) baseado na herança é mais bem aplicado a uma classe base e em classes derivadas
que adicionam outros membros à classe base, ou que exigem funcionalidades adicionais não incluídas na classe
base.
Criação da classe base e das classes derivadas
Vamos examinar o processo de criação de uma classe base e de suas classes derivadas. Nesta seção, você definirá
uma classe base, Publication , que representa uma publicação de qualquer tipo, como um livro, uma revista, um
jornal, um diário, um artigo, etc. Você também definirá uma classe Book que deriva de Publication . É possível
estender facilmente o exemplo para definir outras classes derivadas, como Magazine , Journal , Newspaper e
Article .
Para o seu exemplo, você usará a hierarquia pequena de uma classe Publication e uma única classe
derivada, a Book . É possível ampliar o exemplo facilmente para criar várias classes adicionais derivadas de
Publication , como Magazine e Article .
Se faz sentido instanciar a classe base. Se não fizer, você deverá aplicar a palavra-chave abstract à classe.
Caso contrário, a instância da classe Publication poderá ser criada chamando seu construtor de classe. Se
for feita uma tentativa de instanciar uma classe marcada com a palavra-chave abstract por uma chamada
direta para o construtor de classe, o compilador de C# gerará o erro CS0144, "Não é possível criar uma
instância da classe abstrata ou interface". Se for feita uma tentativa de instanciar a classe por meio da
reflexão, o método de reflexão lançará um MemberAccessException.
Por padrão, uma classe base pode ser instanciada chamando seu construtor de classe. Você não precisa
definir um construtor de classe explicitamente. Se não houver um presente no código-fonte da classe base,
o compilador de C# fornecerá automaticamente um construtor (sem parâmetros) padrão.
No seu exemplo, você marcará a classe Publication como abstract, para que não seja possível criar uma
instância dela. Uma classe abstract sem nenhum método abstract indica que essa classe representa um
conceito abstrato que é compartilhado entre várias classes concretas (como Book , Journal ).
Se as classes derivadas precisam herdar a implementação de membros específicos da classe base, se elas
têm a opção de substituir a implementação da classe base ou se precisam fornecer uma implementação.
Use a palavra-chave abstract para forçar as classes derivadas a fornecer uma implementação. Use a
palavra-chave virtual para permitir que as classes derivadas substituam um método de classe base. Por
padrão, os métodos definidos na classe base não são substituíveis.
A classe Publication não tem nenhum método abstract , mas a classe em si é abstract .
Se uma classe derivada representa a classe final na hierarquia de herança e não pode se ser usada como
uma classe base para outras classes derivadas. Por padrão, qualquer classe pode servir como classe base.
Você pode aplicar a palavra-chave sealed para indicar que uma classe não pode funcionar como uma classe
base para quaisquer classes adicionais. A tentativa de derivar de uma classe selada gerou o erro do
compilador CS0509, "Não é possível derivar do tipo selado <typeName>".
No seu exemplo, você marcará a classe derivada como sealed .
O exemplo a seguir mostra o código-fonte para a classe Publication , bem como uma enumeração
PublicationType que é retornada pela propriedade Publication.PublicationType . Além dos membros herdados de
Object, a classe Publication define os seguintes membros exclusivos e substituições de membro:
using System;
if (title == null)
throw new ArgumentNullException("The title cannot be null.");
else if (String.IsNullOrWhiteSpace(title))
throw new ArgumentException("The title cannot consist only of white space.");
Title = title;
Type = type;
}
Um construtor
Como a classe Publication é abstract , sua instância não pode ser criada diretamente no código, com no
exemplo a seguir:
No entanto, o construtor de instância pode ser chamado diretamente dos construtores de classe derivada,
como mostra o código-fonte para a classe Book .
Duas propriedades relacionadas à publicação
Title é uma propriedade String somente leitura cujo valor é fornecido pela chamada do construtor
Publication .
Pages é uma propriedade Int32 de leitura-gravação que indica o número total de páginas da publicação. O
valor é armazenado em um campo privado chamado totalPages . Ele deve ser um número positivo, caso
contrário, será gerada uma exceção do tipo ArgumentOutOfRangeException.
Membros relacionados ao publicador
Duas propriedades somente leitura, Publisher e Type . Originalmente, os valores eram fornecidos pela
chamada para o construtor da classe Publication .
Membros relacionados à publicação
Dois métodos, Publish e GetPublicationDate , definem e retornam a data de publicação. O método
Publish define um sinalizador published privado como true quando ele é chamado, e atribui a data
passada para ele como um argumento para o campo datePublished privado. O método
GetPublicationDate retorna a cadeia de caracteres "NYP" se o sinalizador published for false , e o valor
do campo datePublished for true .
Membros relacionados a direitos autorais
O método Copyright usa o nome do proprietário dos direitos autorais e o ano dos direitos autorais como
argumentos e os atribui às propriedades CopyrightName e CopyrightDate .
Uma substituição do método ToString
Se um tipo não substituir o método Object.ToString, ele retornará o nome totalmente qualificado do tipo,
que é de pouca utilidade na diferenciação de uma instância para outra. A classe Publication substitui
Object.ToString para retornar o valor da propriedade Title .
A figura a seguir ilustra o relacionamento entre a classe base Publication e sua classe herdada implicitamente
Object.
A classe Book
A classe Book representa um livro como tipo especializado de publicação. O exemplo a seguir mostra o código-
fonte para a classe Book .
using System;
public Book(string title, string isbn, string author, string publisher) : base(title, publisher,
PublicationType.Book)
{
// isbn argument must be a 10- or 13-character numeric string without "-" characters.
// We could also determine whether the ISBN is valid by comparing its checksum digit
// with a computed checksum.
//
if (! String.IsNullOrEmpty(isbn)) {
// Determine if ISBN length is correct.
if (! (isbn.Length == 10 | isbn.Length == 13))
throw new ArgumentException("The ISBN must be a 10- or 13-character numeric string.");
ulong nISBN = 0;
if (! UInt64.TryParse(isbn, out nISBN))
throw new ArgumentException("The ISBN can consist of numeric characters only.");
}
ISBN = isbn;
Author = author;
}
if (currency.Length != 3)
throw new ArgumentException("The ISO currency symbol is a 3-character string.");
Currency = currency;
return oldValue;
}
public override string ToString() => $"{(String.IsNullOrEmpty(Author) ? "" : Author + ", ")}{Title}";
}
Além dos membros herdados de Publication , a classe Book define os seguintes membros exclusivos e
substituições de membro:
Dois construtores
Os dois construtores Book compartilham três parâmetros comuns. Dois, title e publisher, correspondem
aos parâmetros do construtor Publication . O terceiro é author, que é armazenado em uma propriedade
pública Author imutável. Um construtor inclui um parâmetro isbn, que é armazenado na propriedade
automática ISBN .
O primeiro construtor usa a palavra-chave this para chamar o outro construtor. O encadeamento do
construtor é um padrão comum na definição de construtores. Construtores com menos parâmetros
fornecem valores padrão ao chamar o construtor com o maior número de parâmetros.
O segundo construtor usa a palavra-chave base para passar o título e o nome do publicador para o
construtor da classe base. Se você não fizer uma chamada explícita para um construtor de classe base em
seu código-fonte, o compilador de C# fornecerá automaticamente uma chamada para a classe base padrão
ou para o construtor sem parâmetros.
Uma propriedade ISBN somente leitura, que retorna o ISBN (International Standard Book Number) do
objeto Book , um número exclusivo com 10 ou 13 dígitos. O ISBN é fornecido como um argumento para
um dos construtores Book . O ISBN é armazenado em um campo de suporte particular, gerado
automaticamente pelo compilador.
Uma propriedade Author somente leitura. O nome do autor é fornecido como um argumento para os dois
construtores Book e é armazenado na propriedade.
Duas propriedades somente leitura relacionadas ao preço, Price e Currency . Seus valores são fornecidos
como argumentos em uma chamada do método SetPrice . A propriedade Currency é o símbolo de moeda
ISO de três dígitos (por exemplo, USD para dólar americano). Símbolos de moeda ISO podem ser obtidos
da propriedade ISOCurrencySymbol. Ambas as propriedades são somente leitura externamente, mas
podem ser definidas por código na classe Book .
Um método SetPrice , que define os valores das propriedades Price e Currency . Esses valores são
retornados por essas mesmas propriedades.
Substitui o método ToString (herdado de Publication ) e os métodos Object.Equals(Object) e
GetHashCode (herdados de Object).
A menos que seja substituído, o método Object.Equals(Object) testa a igualdade de referência. Ou seja,
duas variáveis de objeto são consideradas iguais se fizerem referência ao mesmo objeto. Na classe Book ,
por outro lado, dois objetos Book devem ser iguais quando têm o mesmo ISBN.
Ao substituir o método Object.Equals(Object), substitua também o método GetHashCode, que retorna um
valor usado pelo tempo de execução para armazenar itens em coleções de hash para uma recuperação
eficiente. O código de hash deve retornar um valor que é consistente com o teste de igualdade. Como você
substituiu Object.Equals(Object) para retornar true , se as propriedades de ISBN de dois objetos Book
forem iguais, retorne o código hash computado chamando o método GetHashCode da cadeia de caracteres
retornada pela propriedade ISBN .
A figura a seguir ilustra o relacionamento entre a classe Book e Publication , sua classe base.
Agora você pode criar a instância de um objeto Book , invocar seus membros exclusivos e herdados e passá-lo
como um argumento a um método que espera um parâmetro do tipo Publication ou do tipo Book , como mostra
o exemplo a seguir.
using System;
using static System.Console;
var book2 = new Book("The Tempest", "Classic Works Press", "Shakespeare, William");
Write($"{book.Title} and {book2.Title} are the same publication: " +
$"{((Publication) book).Equals(book2)}");
}
Por exemplo, cada forma geométrica bidimensional fechada inclui duas propriedades: área, a extensão interna da
forma; e perímetro, ou a distância entre as bordas da forma. A maneira com a qual essas propriedades são
calculadas, no entanto, depende completamente da forma específica. A fórmula para calcular o perímetro (ou a
circunferência) de um círculo, por exemplo, é diferente da fórmula de um triângulo. A classe Shape é uma classe
abstract com métodos abstract . Isso indica que as classes derivadas compartilham a mesma funcionalidade,
mas essas classes derivadas implementam essa funcionalidade de forma diferente.
O exemplo a seguir define uma classe base abstrata denominada Shape que define duas propriedades: Area e
Perimeter . Além da classe ser marcada com a palavra-chave abstract, cada membro da instância também é
marcado com a palavra-chave abstract. Nesse caso, o Shape também substitui o método Object.ToString para
retornar o nome do tipo, em vez de seu nome totalmente qualificado. E define dois membros estáticos, GetArea e
GetPerimeter , que permitem a recuperação fácil da área e do perímetro de uma instância de qualquer classe
derivada. Quando você passa uma instância de uma classe derivada para um desses métodos, o runtime chama a
substituição do método da classe derivada.
using System;
Em seguida, você pode derivar algumas classes de Shape que representam formas específicas. O exemplo a
seguir define três classes, Triangle , Rectangle e Circle . Cada uma usa uma fórmula exclusiva para essa forma
específica para calcular a área e o perímetro. Algumas das classes derivadas também definem propriedades, como
Rectangle.Diagonal e Circle.Diameter , que são exclusivas para a forma que representam.
using System;
O exemplo a seguir usa objetos derivados de Shape . Ele cria uma matriz de objetos derivados de Shape e chama
os métodos estáticos da classe Shape , que retorna valores de propriedade Shape . O runtime recupera os valores
das propriedades substituídas dos tipos derivados. O exemplo também converte cada objeto Shape na matriz ao
seu tipo derivado e, se a conversão for bem-sucedida, recupera as propriedades dessa subclasse específica de
Shape .
using System;
Consulte também
Classes e objetos
Herança (Guia de programação em C#)
Trabalhando com LINQ
30/10/2019 • 28 minutes to read • Edit Online
Introdução
Este tutorial ensina os recursos no .NET Core e da linguagem C#. Você aprenderá:
Como gerar sequências com LINQ.
Como escrever métodos que podem ser facilmente usados em consultas LINQ.
Como distinguir entre uma avaliação lenta e uma detalhada.
Você aprenderá essas técnicas ao compilar um aplicativo que demonstra uma das habilidades básicas de qualquer
mágico: o embaralhamento faro. Em resumo, um embaralhamento faro é uma técnica em que você divide um
baralho de cartas exatamente na metade, então as cartas de cada metade são colocadas em ordem aleatória até
recriar o conjunto original.
Os mágicos usam essa técnica porque cada carta é fica em um local conhecido após o embaralhamento e a ordem
é um padrão de repetição.
Para os seus propósitos, vamos examinar rapidamente as sequências de manipulação de dados. O aplicativo que
você criará construirá um baralho de cartas e, em seguida, executará uma sequência de embaralhamento, sempre
gravando a sequência de saída. Você também comparará a ordem atualizada com a ordem original.
Este tutorial tem várias etapas. Após cada etapa, você poderá executar o aplicativo e ver o progresso. Você
também poderá ver o exemplo concluído no repositório dotnet/samples do GitHub. Para obter instruções de
download, consulte Exemplos e tutoriais.
Prerequisites
Você precisará configurar seu computador para executar o .NET Core. Você pode encontrar as instruções de
instalação na página de download do .NET Core . Você pode executar esse aplicativo no Windows, Ubuntu Linux,
OS X ou em um contêiner do Docker. Você precisará instalar o editor de código de sua preferência. As descrições
a seguir usam o Visual Studio Code, que é uma software livre, no editor de plataforma. No entanto, você pode
usar quaisquer ferramentas que esteja familiarizado.
Criar o aplicativo
A primeira etapa é criar um novo aplicativo. Abra um prompt de comando e crie um novo diretório para seu
aplicativo. Torne ele o diretório atual. Digite o comando dotnet new console no prompt de comando. Isso cria os
arquivos iniciais de um aplicativo "Olá, Mundo" básico.
Se você nunca usou C# antes, este tutorial explicará a estrutura de um programa C#. Você pode ler e, em seguida,
voltar aqui para saber mais sobre o LINQ.
Se essas três linhas (instruções using ) não estiverem na parte superior do arquivo, nosso programa não será
compilado.
Agora que você tem todas as referências necessárias, considere o que forma um baralho de cartas. Um baralho de
cartas costuma ter quatro naipes, e cada naipe tem treze valores. Normalmente, talvez você pense em criar uma
classe Card logo de cara e preencher uma coleção de objetos Card manualmente. Com o LINQ, dá para ser mais
conciso do que a forma comum de criação de um baralho de cartas. Em vez de criar uma classe Card , você pode
criar duas sequências para representar naipes e valores, respectivamente. Você vai criar um par muito simples de
métodos iteradores que gerará as valores e naipes como IEnumerable<T>s de cadeias de caracteres:
// Program.cs
// The Main() method
Coloque-as sob o método Main em seu arquivo Program.cs . Esses dois métodos utilizam a sintaxe yield return
para produzir uma sequência à medida que eles são executados. O compilador compila um objeto que
implementa IEnumerable<T> e gera a sequência de cadeias de caracteres conforme solicitado.
Agora, use esses métodos iteradores para criar o baralho de cartas. Você colocará a consulta do LINQ em nosso
método Main . Dê uma olhada:
// Program.cs
static void Main(string[] args)
{
var startingDeck = from s in Suits()
from r in Ranks()
select new { Suit = s, Rank = r };
// Display each card that we've generated and placed in startingDeck in the console
foreach (var card in startingDeck)
{
Console.WriteLine(card);
}
}
As várias cláusulas from produzem um SelectMany, que cria uma única sequência da combinação entre cada
elemento na primeira sequência com cada elemento na segunda sequência. A ordem é importante para nossos
objetivos. O primeiro elemento na primeira sequência de fonte (Naipes) é combinado com cada elemento na
segunda sequência (Valores). Isso produz todas as treze cartas do primeiro naipe. Esse processo é repetido com
cada elemento na primeira sequência (naipes). O resultado final é um baralho ordenado por naipes, seguido pelos
valores.
É importante lembrar que se você optar por escrever seu LINQ na sintaxe de consulta usada acima, ou se decidir
usar a sintaxe de método, sempre será possível alternar entre as formas de sintaxe. A consulta acima escrita em
sintaxe de consulta pode ser escrita na sintaxe de método como:
var startingDeck = Suits().SelectMany(suit => Ranks().Select(rank => new { Suit = suit, Rank = rank }));
O compilador traduz instruções LINQ escritas com a sintaxe de consulta na sintaxe de chamada do método
equivalente. Portanto, independentemente de sua escolha de sintaxe, as duas versões da consulta produzem o
mesmo resultado. Escolha qual sintaxe funciona melhor para a sua situação: por exemplo, se você estiver
trabalhando em uma equipe em que alguns dos membros têm dificuldade com a sintaxe de método, prefira usar a
sintaxe de consulta.
Vá em frente e execute o exemplo que você criou neste momento. Ele exibirá todas as 52 cartas do baralho. Talvez
seja muito útil executar esse exemplo em um depurador para observar como os métodos Suits() e Ranks() são
executados. Você pode ver claramente que cada cadeia de caracteres em cada sequência é gerada apenas
conforme o necessário.
Manipulando a ordem
Em seguida, concentre-se em como você vai embaralhar as cartas no baralho. A primeira etapa de qualquer
embaralhada é dividir o baralho em dois. Os métodos Take e Skip que fazem parte das APIs do LINQ fornecem
esse recurso para você. Coloque-os sob o loop foreach :
// Program.cs
public static void Main(string[] args)
{
var startingDeck = from s in Suits()
from r in Ranks()
select new { Suit = s, Rank = r };
// 52 cards in a deck, so 52 / 2 = 26
var top = startingDeck.Take(26);
var bottom = startingDeck.Skip(26);
}
No entanto, não há método de embaralhamento na biblioteca padrão, portanto, você precisará escrever o seu. O
método de embaralhamento que você criará ilustra várias técnicas que você usará com programas baseados em
LINQ. Portanto, cada parte desse processo será explicado nas etapas.
Para adicionar funcionalidade ao seu modo de interação com o IEnumerable<T> recebido de volta das consultas
do LINQ, precisará escrever alguns tipos especiais de métodos chamados métodos de extensão. Em resumo, um
método de extensão é um método estático de objetivo especial que adiciona novas funcionalidades a um tipo já
existentes, sem ter que modificar o tipo original ao qual você deseja adicionar funcionalidade.
Dê aos seus métodos de extensão uma nova casa adicionando um novo arquivo de classe estático ao seu
programa chamado Extensions.cs , depois, comece a criar o primeiro método de extensão:
// Extensions.cs
using System;
using System.Collections.Generic;
using System.Linq;
namespace LinqFaroShuffle
{
public static class Extensions
{
public static IEnumerable<T> InterleaveSequenceWith<T>(this IEnumerable<T> first, IEnumerable<T>
second)
{
// Your implementation will go here soon enough
}
}
}
Você pode ver a adição do modificador this no primeiro argumento para o método. Isso significa que você
chama o método como se fosse um método de membro do tipo do primeiro argumento. Esta declaração de
método também segue um idioma padrão no qual os tipos de entrada e saídas são IEnumerable<T> . Essa prática
permite que os métodos LINQ sejam encadeados para executar consultas mais complexas.
Naturalmente, como você dividiu o baralho em metades, precisará unir essas metades. No código, isso significa
que você vai enumerar as duas sequências adquiridas por meio de Take e Skip ao mesmo tempo, interleaving os
elementos e criará uma sequência: seu baralho não embaralhado. Escrever um método LINQ que funciona com
duas sequências exige que você compreenda como IEnumerable<T> funciona.
A interface IEnumerable<T> tem um método: GetEnumerator. O objeto retornado por GetEnumerator tem um
método para mover para o próximo elemento e uma propriedade que recupera o elemento atual na sequência.
Você usará esses dois membros para enumerar a coleção e retornar os elementos. Esse método de Intercalação
será um método iterador, portanto, em vez de criar uma coleção e retornar a coleção, você usará a sintaxe
yield return mostrada acima.
Agora que você escreveu esse método, vá até o método Main e embaralhe uma vez:
// Program.cs
public static void Main(string[] args)
{
var startingDeck = from s in Suits()
from r in Ranks()
select new { Suit = s, Rank = r };
Comparações
Quantos embaralhamentos são necessários para colocar o baralho em sua ordem original? Para descobrir, você
precisará escrever um método que determina se duas sequências são iguais. Depois de ter esse método, você
precisará colocar o código de embaralhamento em um loop e verificar quando a apresentação estiver na ordem.
Escrever um método para determinar se as duas sequências são iguais deve ser simples. É uma estrutura
semelhante para o método que você escreveu para embaralhar as cartas. Somente desta vez, em vez de o
yield return rendimento retornar cada elemento, você comparará os elementos correspondentes de cada
sequência. Quando toda a sequência tiver sido enumerada, se os elementos corresponderem, as sequências serão
as mesmas:
return true;
}
Isso mostra uma segunda linguagem LINQ: métodos de terminal. Eles consideram uma sequência como entrada
(ou, neste caso, duas sequências) e retornam um único valor escalar. Ao usar métodos de terminal, eles são
sempre o método final em uma cadeia de métodos para uma consulta LINQ, por isso, o nome "terminal".
Você pode ver isso em ação ao usá-lo para determinar quando o baralho está em sua ordem original. Coloque o
código de embaralhamento dentro de um loop e pare quando a sequência estiver em sua ordem original,
aplicando o método SequenceEquals() . Você pode ver que esse sempre será o método final em qualquer consulta,
porque ele retorna um valor único em vez de uma sequência:
// Program.cs
static void Main(string[] args)
{
// Query for building the deck
var times = 0;
// We can re-use the shuffle variable from earlier, or you can make a new one
shuffle = startingDeck;
do
{
shuffle = shuffle.Take(26).InterleaveSequenceWith(shuffle.Skip(26));
} while (!startingDeck.SequenceEquals(shuffle));
Console.WriteLine(times);
}
Execute o código que obtivemos até agora e observe como o baralho é reorganizado em cada embaralhamento.
Após 8 embaralhamentos (iterações do loop do-while), o baralho retorna à configuração original que estava
quando você o criou pela primeira vez a partir da consulta LINQ inicial.
Otimizações
O exemplo que você compilou até agora executa um embaralhamento externo, no qual as cartas superiores e
inferiores permanecem as mesmas em cada execução. Vamos fazer uma alteração: em vez disso, usaremos um
embaralhamento interno, em que todas as 52 cartas trocam de posição. Para um embaralhamento interno,
intercale o baralho para que a primeira carta da metade inferior torne-se a primeira carta do baralho. Isso
significa que a última carta na metade superior torna-se a carta inferior. Essa é uma alteração simples em uma
única linha de código. Atualize a consulta atual de embaralhamento, alternando as posições de Take e Skip. Isso
alterará a ordem das metades superior e inferior do baralho:
shuffle = shuffle.Skip(26).InterleaveSequenceWith(shuffle.Take(26));
Execute o programa novamente e você verá que leva 52 iterações para o baralho ser reordenado. Você também
começará a observar algumas degradações de desempenho graves à medida que o programa continuar a ser
executado.
Existem muitas razões para isso. Você pode abordar uma das principais causas dessa queda de desempenho: uso
ineficiente da avaliação lenta.
Em resumo, a avaliação lenta informa que a avaliação de uma instrução não será executada até que seu valor seja
necessário. Consultas LINQ são instruções avaliadas lentamente. As sequências são geradas somente quando os
elementos são solicitados. Geralmente, esse é o principal benefício do LINQ. No entanto, em uso como esse
programa, isso causa um crescimento exponencial no tempo de execução.
Lembre-se de que geramos o baralho original usando uma consulta LINQ. Cada embaralhamento é gerado
executando três consultas LINQ no baralho anterior. Todos eles são executados lentamente. Isso também significa
que eles são executados novamente sempre que a sequência é solicitada. Ao obter a 52ª iteração, você estará
regenerando o baralho original muitas e muitas vezes. Vamos escrever um log para demonstrar esse
comportamento. Em seguida, você poderá corrigir isso.
Em seu arquivo Extensions.cs , digite ou copie o método a seguir. Esse método de extensão cria um novo arquivo
chamado debug.log em seu diretório do projeto, e registra qual consulta está sendo executada atualmente para o
arquivo de log. Este método de extensão pode ser anexado a qualquer consulta para marcar que a consulta foi
executada.
return sequence;
}
Você verá um rabisco vermelho sob File , que significa que ele não existe. Ele não será compilado, pois o
compilador não sabe o que é File . Para resolver esse problema, é preciso que você adicione a linha de código a
seguir abaixo da primeira linha em Extensions.cs :
using System.IO;
Console.WriteLine();
var times = 0;
var shuffle = startingDeck;
do
{
// Out shuffle
/*
shuffle = shuffle.Take(26)
.LogQuery("Top Half")
.InterleaveSequenceWith(shuffle.Skip(26)
.LogQuery("Bottom Half"))
.LogQuery("Shuffle");
*/
// In shuffle
shuffle = shuffle.Skip(26).LogQuery("Bottom Half")
.InterleaveSequenceWith(shuffle.Take(26).LogQuery("Top Half"))
.LogQuery("Shuffle");
times++;
Console.WriteLine(times);
} while (!startingDeck.SequenceEquals(shuffle));
Console.WriteLine(times);
}
Observe que você não precisa fazer o registro sempre que acessar uma consulta. Você faz o registro ao criar a
consulta original. O programa ainda leva muito tempo para ser executado, mas agora você pode ver o motivo. Se
você não tiver paciência para executar o embaralhamento interno com o registro em log ativado, volte para o
embaralhamento externo. Você ainda verá os efeitos da avaliação lenta. Em uma execução, ele faz 2592 consultas,
incluindo a geração de todos os valores e naipes.
Aqui, você pode melhorar o desempenho do código para reduzir o número de execuções feitas. Uma correção
simples possível é armazenar em cache os resultados da consulta do LINQ original que constrói o baralho de
cartas. Atualmente, você executa as consultas novamente sempre que o loop do-while passa por uma iteração,
construindo novamente o baralho de cartas e o embaralhamento de novo todas as vezes. Para armazenar em
cache o baralho de cartas, aproveite os métodos LINQ ToArray e ToList; ao anexá-los às consultas, eles executarão
as mesmas ações paras quais foram instruídos, mas agora armazenarão os resultados em uma matriz ou lista,
dependendo de qual método você optar por chamar. Anexe o método LINQ ToArray às duas consultas e execute o
programa novamente:
public static void Main(string[] args)
{
var startingDeck = (from s in Suits().LogQuery("Suit Generation")
from r in Ranks().LogQuery("Value Generation")
select new { Suit = s, Rank = r })
.LogQuery("Starting Deck")
.ToArray();
Console.WriteLine();
var times = 0;
var shuffle = startingDeck;
do
{
/*
shuffle = shuffle.Take(26)
.LogQuery("Top Half")
.InterleaveSequenceWith(shuffle.Skip(26).LogQuery("Bottom Half"))
.LogQuery("Shuffle")
.ToArray();
*/
shuffle = shuffle.Skip(26)
.LogQuery("Bottom Half")
.InterleaveSequenceWith(shuffle.Take(26).LogQuery("Top Half"))
.LogQuery("Shuffle")
.ToArray();
times++;
Console.WriteLine(times);
} while (!startingDeck.SequenceEquals(shuffle));
Console.WriteLine(times);
}
Agora, o embaralhamento externo contém 30 consultas. Execute novamente com o embaralhamento interno e
você verá melhorias semelhantes: agora, executa 162 consultas.
Observe que esse exemplo é projetado para realçar os casos de uso em que a avaliação lenta pode causar
problemas de desempenho. Embora seja importante ver onde a avaliação lenta pode afetar o desempenho do
código, é igualmente importante entender que nem todas as consultas devem ser executadas avidamente. O
desempenho incorrido sem usar ToArray ocorre porque cada nova disposição do baralho de cartas é criada com
base na disposição anterior. Usar a avaliação lenta significa que cada nova disposição do baralho é criada do
baralho original, até mesmo a execução do código que criou o startingDeck . Isso causa uma grande quantidade
de trabalho extra.
Na prática, alguns algoritmos funcionam bem usando a avaliação detalhada, e outros executam funcionam melhor
usando a avaliação lenta. Para o uso diário, a avaliação lenta é uma opção melhor quando a fonte de dados é um
processo separado, como um mecanismo de banco de dados. Para os bancos de dados, a avaliação lenta permite
que as consultas mais complexas executem apenas uma viagem de ida e volta para o processo de banco de dados
e de volta para o restante do seu código. O LINQ é flexível, não importa se você optar por utilizar a avaliação lenta
ou detalhada, portanto, meça seus processos e escolha o tipo de avaliação que ofereça o melhor desempenho.
Conclusão
Neste projeto, abordamos:
o uso de consultas LINQ para agregar dados em uma sequência significativa
a produção de métodos de Extensão para adicionar nossa própria funcionalidade personalizada a consultas
LINQ
a localização de áreas em nosso código nas quais nossas consultas LINQ podem enfrentar problemas de
desempenho, como diminuição da velocidade
avaliação lenta e detalhada com relação às consultas LINQ, e as implicações que elas podem ter no
desempenho da consulta
Além do LINQ, você aprendeu um pouco sobre uma técnica usada por mágicos para truques de carta. Os
mágicos usam o embaralhamento Faro porque podem controlar onde cada carta fica no baralho. Agora que você
sabe, não conte para os outros!
Para saber mais sobre o LINQ, consulte:
LINQ (Consulta Integrada à Linguagem)
Introdução ao LINQ
Operações de consulta LINQ básica (C#)
Transformações de dados com LINQ (C#)
Sintaxe de consulta e sintaxe de método em LINQ (C#)
Recursos do C# que dão suporte a LINQ
Usando atributos em C#
30/10/2019 • 14 minutes to read • Edit Online
Os atributos fornecem uma maneira de associar informações ao código de forma declarativa. Eles também podem
fornecer um elemento reutilizável que pode ser aplicado a uma variedade de destinos.
Considere o atributo [Obsolete] . Ele pode ser aplicado a classes, structs, métodos, construtores e muito mais. Ele
declara que o elemento é obsoleto. Em seguida, cabe ao compilador C# procurar esse atributo e realizar alguma
ação em resposta.
Neste tutorial, você verá como adicionar atributos a seu código, como criar e usar seus próprios atributos e como
usar alguns atributos que são criados no .NET Core.
Prerequisites
Você precisará configurar seu computador para executar o .NET Core. Você pode encontrar as instruções de
instalação na página de downloads do .NET Core . Você pode executar esse aplicativo no Windows, Ubuntu Linux,
macOS ou em um contêiner do Docker. Você precisará instalar o editor de código de sua preferência. As
descrições a seguir usam o Visual Studio Code, que é uma software livre, no editor de plataforma. No entanto,
você pode usar quaisquer ferramentas que esteja familiarizado.
Criar o aplicativo
Agora que você instalou todas as ferramentas, crie um novo aplicativo do .NET Core. Para usar o gerador de linha
de comando, execute o seguinte comando no shell de sua preferência:
dotnet new console
Esse comando criará arquivos de projeto do .NET Core barebones. Você precisará executar dotnet restore para
restaurar as dependências necessárias para compilar esse projeto.
NOTE
Começando com o SDK do .NET Core 2.0, não é necessário executar dotnet restore , pois ele é executado implicitamente
por todos os comandos que exigem uma restauração, como dotnet new , dotnet build e dotnet run . Ainda é um
comando válido em determinados cenários em que realizar uma restauração explícita faz sentido, como builds de integração
contínua no Azure DevOps Services ou em sistemas de build que precisam controlar explicitamente o horário em que a
restauração ocorrerá.
Para executar o programa, use dotnet run . Você deve ver a saída do "Olá, Mundo" no console.
Observe que, embora a classe seja chamada de ObsoleteAttribute , só é necessário usar [Obsolete] no código.
Isso é uma convenção que a linguagem C# segue. Você poderá usar o nome completo [ObsoleteAttribute] se
escolher.
Ao marcar uma classe obsoleta, é uma boa ideia fornecer algumas informações como o motivo de estar obsoleto
e/ou o que usar no lugar. Faça isso passando um parâmetro de cadeia de caracteres para o atributo obsoleto.
A cadeia de caracteres está sendo passada como um argumento para um construtor ObsoleteAttribute , como se
você estivesse escrevendo var attr = new ObsoleteAttribute("some string") .
Os parâmetros para um construtor de atributo são limitados a literais/tipos simples:
bool, int, double, string, Type, enums, etc e matrizes desses tipos. Você não pode usar uma expressão ou uma
variável. Você pode usar parâmetros posicionais ou nomeados.
Com os itens acima, agora posso usar [MySpecial] (ou [MySpecialAttribute] ) como um atributo em qualquer
lugar na base do código.
[MySpecial]
public class SomeOtherClass
{
}
}
No entanto, não será possível usar esse construtor com a sintaxe de atributo.
[AttributeUsage(AttributeTargets.Class | AttributeTargets.Struct)]
public class MyAttributeForClassAndStructOnly : Attribute
{
Se tentar colocar o atributo acima em algo que não é uma classe ou um struct, você obterá um erro do compilador
como
Attribute 'MyAttributeForClassAndStructOnly' is not valid on this declaration type. It is only valid on 'class,
struct' declarations
public class Foo
{
// if the below attribute was uncommented, it would cause a compiler error
// [MyAttributeForClassAndStructOnly]
public Foo()
{ }
}
Depois de ter um objeto TypeInfo (ou um MemberInfo , FieldInfo , etc.), você poderá usar o método
GetCustomAttributes . Isso retornará uma coleção de objetos Attribute . Você também poderá usar
GetCustomAttribute e especificar um tipo de atributo.
Aqui está um exemplo do uso de GetCustomAttributes em uma instância MemberInfo para MyClass (que vimos,
anteriormente, que tem um atributo [Obsolete] nele).
Isso será impresso no console: Attribute on MyClass: ObsoleteAttribute . Tente adicionar outros atributos a
MyClass .
É importante observar que esses objetos Attribute são instanciados lentamente. Ou seja, eles não serão
instanciados até que você use GetCustomAttribute ou GetCustomAttributes . Eles também são instanciados a cada
vez. Chamar GetCustomAttributes duas vezes em uma linha retornará duas instâncias diferentes do
ObsoleteAttribute .
No código acima, você não precisa ter uma cadeia de caracteres "Name" literal. Isso pode ajudar a evitar erros
relacionados à digitação e também possibilita a refatoração/renomeação mais suave.
Resumo
Atributos trazem poder declarativo para C#, mas eles são uma forma de metadados de código e não agem
sozinhos.
Um tour pela linguagem C#
24/10/2019 • 9 minutes to read • Edit Online
O C# (pronuncia-se "C Sharp") é uma linguagem de programação simples, moderna, orientada a objeto e
fortemente tipada. O C# tem suas raízes na família de linguagens C e os programadores em C, C++, Java e
JavaScript a reconhecerão imediatamente. Este tour dá uma visão geral dos principais componentes da linguagem.
Se quiser explorar a linguagem por meio de exemplos interativos, experimente nossos tutoriais de Introdução à
linguagem C#.
O C# é uma linguagem orientada a objeto, mas inclui ainda suporte para programação orientada a componentes.
O design de software atual depende cada vez mais dos componentes de software na forma de pacotes
independentes e autodescritivos de funcionalidade. O principal é que esses componentes apresentam um modelo
de programação com propriedades, métodos e eventos; eles têm atributos que fornecem informações declarativas
sobre o componente; e incorporam sua própria documentação. C# fornece construções de linguagem para dar
suporte diretamente a esses conceitos, tornando C# uma linguagem muito natural para criação e uso de
componentes de software.
Vários recursos do C# auxiliam na construção de aplicativos robustos e duradouros: Coleta de lixo recupera
automaticamente a memória ocupada por objetos inacessíveis não utilizados; tratamento de exceção fornece uma
abordagem estruturada e extensível para detecção e recuperação de erros; e o design fortemente tipado da
linguagem impossibilita a leitura das variáveis não inicializadas, a indexação de matrizes além dos seus limites ou a
execução de conversões de tipo não verificadas.
C# tem um sistema de tipo unificado. Todos os tipos do C#, incluindo tipos primitivos, como int e double ,
herdam de um único tipo de object raiz. Assim, todos os tipos compartilham um conjunto de operações comuns,
e valores de qualquer tipo podem ser armazenados, transportados e operados de maneira consistente. Além disso,
C# oferece suporte a tipos de referência e tipos de valor definidos pelo usuário, permitindo a alocação dinâmica de
objetos, bem como o armazenamento em linha de estruturas leves.
Para garantir que C# os programas e as bibliotecas possam evoluir ao longo do tempo de maneira compatível,
muito ênfase foi colocado no C#design do controle de versão . Muitas linguagens de programação prestam pouca
atenção a esse problema e, como resultado, programas escritos nessas linguagens quebram com mais frequência
do que o necessário quando versões mais recentes das bibliotecas dependentes são introduzidas. Aspectos do
C#design de do que foram influenciados diretamente pelas considerações de controle de versão incluem os
modificadores virtual e override separados, as regras para resolução de sobrecarga de método e suporte para
declarações de membro de interface explícitas.
Hello world
O programa "Hello, World" é usado tradicionalmente para introduzir uma linguagem de programação. Este é para
C#:
using System;
class Hello
{
static void Main()
{
Console.WriteLine("Hello, World");
}
}
Os arquivos de origem em C# normalmente têm a extensão de arquivo .cs . Supondo que o programa "Olá,
mundo" esteja armazenado no arquivo Hello.cs, o programa poderá ser compilado usando a linha de comando:
csc hello.cs
que produz um assembly executável chamado Hello. exe. O resultado produzido pela execução desse aplicativo é:
Hello, World
IMPORTANT
O comando csc compila para o framework completo e talvez não esteja disponível em todas as plataformas.
O programa "Hello, World" começa com uma diretiva using que faz referência ao namespace System .
Namespaces fornecem um meio hierárquico de organizar bibliotecas e programas em C#. Os namespaces contêm
tipos e outros namespaces — por exemplo, o namespace System contém uma quantidade de tipos, como a classe
Console referenciada no programa e diversos outros namespaces, como IO e Collections . A diretiva using que
faz referência a um determinado namespace permite o uso não qualificado dos tipos que são membros desse
namespace. Devido à diretiva using , o programa pode usar Console.WriteLine como um atalho para
System.Console.WriteLine .
A classe Hello declarada pelo programa "Hello, World" tem um único membro, o método chamado Main . O
método Main é declarado com o modificador estático. Embora os métodos de instância possam fazer referência a
uma determinada instância de objeto delimitador usando a palavra-chave this , métodos estáticos operam sem
referência a um objeto específico. Por convenção, um método estático denominado Main serve como ponto de
entrada de um programa.
A saída do programa é produzida pelo método WriteLine da classe Console no namespace System . Essa classe é
fornecida pelas bibliotecas de classe padrão, que, por padrão, são referenciadas automaticamente pelo compilador.
Há muito mais para aprender sobre C#. Os tópicos a seguir fornecem uma visão geral dos elementos da
linguagem C#. Essas visões gerais fornecerão informações básicas sobre todos os elementos da linguagem e
fornecerão as informações necessárias para se aprofundar nos elementos da linguagem C#:
Estrutura do programa
Aprenda os principais conceitos organizacionais na linguagem C#: programas, namespaces, tipos,
membros e assemblies.
Tipos e variáveis
Saiba mais sobre tipos de valor, tipos de referência, e variáveis na linguagem C#.
Expressões
Expressões são construídas a partir de operandos e operadores. As expressões produzem um valor.
Instruções
Você usa instruções para expressar as ações de um programa.
Classes e objetos
As classes são os tipos do C# mais fundamentais. Os objetos são instâncias de uma classe. As classes
são compiladas usando membros, que também são abordados neste tópico.
Structs
Os structs são estruturas de dados que, diferentemente das classes, são tipos de valor.
Matrizes
Uma matriz é uma estrutura de dados que contém algumas variáveis acessadas por meio de índices
calculados.
Interfaces
Uma interface define um contrato que pode ser implementado por classes e estruturas. Uma interface
pode conter métodos, propriedades, eventos e indexadores. Uma interface não fornece implementações
dos membros que define — ela simplesmente especifica os membros que devem ser fornecidos por
classes ou estruturas que implementam a interface.
Enums
Um tipo enum é um tipo de valor diferente com um conjunto de constantes nomeadas.
Delegados
Um delegado é um tipo que representa referências aos métodos com uma lista de parâmetros e tipo de
retorno específicos. Delegados possibilitam o tratamento de métodos como entidades que podem ser
atribuídos a variáveis e passadas como parâmetros. Os delegados são parecidos com o conceito de
ponteiros de função em outras linguagens, mas ao contrário dos ponteiros de função, os delegados são
orientados a objetos e fortemente tipados.
Atributos
Atributos permitem que programas especifiquem informações declarativas adicionais sobre tipos,
membros e outras entidades.
A VA N ÇA R
Estrutura do programa
23/10/2019 • 5 minutes to read • Edit Online
using System;
namespace Acme.Collections
{
public class Stack
{
Entry top;
public void Push(object data)
{
top = new Entry(top, data);
}
class Entry
{
public Entry next;
public object data;
public Entry(Entry next, object data)
{
this.next = next;
this.data = data;
}
}
}
}
O nome totalmente qualificado dessa classe é Acme.Collections.Stack . A classe contém vários membros: um
campo chamado top , dois métodos chamados Push e Pop e uma classe aninhada chamada Entry . A classe
Entry ainda contém três membros: um campo chamado next , um campo chamado data e um construtor.
Supondo que o código-fonte do exemplo seja armazenado no arquivo acme.cs , a linha de comando
compila o exemplo como uma biblioteca (o código sem um ponto de entrada Main ) e produz um assembly
denominado acme.dll .
IMPORTANT
Os exemplos acima usam csc como o compilador C# da linha de comando. Esse compilador é um executável do Windows.
Para usar C# em outras plataformas, você deve usar as ferramentas para o .NET Core. O ecossistema do .NET Core usa o CLI
dotnet para gerenciar as compilações de linha de comando. Isso inclui gerenciamento de dependências e invocação do
compilador C#. Consulte este tutorial para obter uma descrição completa dessas ferramentas nas plataformas com suporte
do .NET Core.
using System;
using Acme.Collections;
class Example
{
static void Main()
{
Stack s = new Stack();
s.Push(1);
s.Push(10);
s.Push(100);
Console.WriteLine(s.Pop());
Console.WriteLine(s.Pop());
Console.WriteLine(s.Pop());
}
}
Se o programa é armazenado no arquivo example.cs , quando example.cs é compilado, o assembly acme.dll pode
ser referenciado usando a opção de /r do compilador:
Isso cria um assembly executável denominado example.exe , que, quando executado, produz a saída:
100
10
1
O C# permite que o texto de origem de um programa seja armazenado em vários arquivos de origem. Quando
um programa em C# com vários arquivo é compilado, todos os arquivos de origem são processados juntos e os
arquivos de origem podem referenciar livremente uns aos outros. Conceitualmente, é como se todos os arquivos
de origem fossem concatenados em um arquivo grande antes de serem processados. Declarações de
encaminhamento nunca são necessárias em C#, porque, com poucas exceções, a ordem de declaração é
insignificante. O C# não limita um arquivo de origem para declarar somente um tipo público nem requer o nome
do arquivo de origem para corresponder a um tipo declarado no arquivo de origem.
A N T E R IO R P R Ó X IM O
Tipos e variáveis
08/11/2019 • 12 minutes to read • Edit Online
Há dois tipos em C#: tipos de referência e tipos de valor. As variáveis de tipos de valor contêm diretamente seus
dados enquanto variáveis de tipos de referência armazenam referências a seus dados, o último sendo conhecido
como objetos. Com tipos de referência, é possível que duas variáveis referenciem o mesmo objeto e, portanto, é
possível que operações em uma variável afetem o objeto referenciado por outra variável. Com tipos de valor, cada
variável tem sua própria cópia dos dados e não é possível que operações em uma variável afetem a outra (exceto
no caso de variáveis de parâmetros ref e out ).
Os tipos de valor do C# são divididos em tipos simples, tipos de enum, tipos struct e tipos de valor anulável. Os
tipos de referência do C# são divididos em tipos de classe, tipos de interface, tipos de matriz e tipos delegados.
O exemplo a seguir fornece uma visão geral do sistema de tipos do C#.
Tipos de valor
Tipos simples
Integral com sinal: sbyte , short , int , long
Integral sem sinal: byte , ushort , uint , ulong
Caracteres Unicode: char
Ponto flutuante binário de IEEE: float , double
Ponto flutuante decimal de alta precisão: decimal
Booliano: bool
Tipos enumerados
Tipos definidos pelo usuário do formulário enum E {...}
Tipos struct
Tipos definidos pelo usuário do formulário struct S {...}
Tipos de valor anuláveis
Extensões de todos os outros tipos de valor com um valor null
Tipos de referência
Tipos de classe
Classe base definitiva de todos os outros tipos: object
Cadeia de caracteres Unicode: string
Tipos definidos pelo usuário do formulário class C {...}
Tipos de interface
Tipos definidos pelo usuário do formulário interface I {...}
Tipos de matriz
Unidimensional e multidimensional, por exemplo, int[] e int[,]
Tipos delegados
Tipos definidos pelo usuário do formulário delegate int D(...)
Para obter mais informações sobre tipos numéricos, veja a tabela Tipos integrais e Tipos de ponto flutuante.
O tipo bool do C# é usado para representar valores boolianos — valores que são true ou false .
O processamento de cadeia de caracteres e caracteres em C# usa codificação Unicode. O tipo char representa
uma unidade de código UTF -16 e o tipo string representa uma sequência de unidades de código UTF -16.
Os programas em C# usam declarações de tipos para criar novos tipos. Uma declaração de tipo especifica o
nome e os membros do novo tipo. Cinco das categorias do C# de tipos são tipos definidos pelo usuário: tipos de
classe, tipos struct, tipos de interface, tipos enum e tipos delegados.
Um tipo class define uma estrutura de dados que contém membros de dados (campos) e membros de função
(métodos, propriedades e outros). Os tipos de classe dão suporte à herança única e ao polimorfismo, mecanismos
nos quais as classes derivadas podem estender e especializar as classes base.
Um tipo struct é semelhante a um tipo de classe que representa uma estrutura com membros de dados e
membros da função. No entanto, diferentemente das classes, structs são tipos de valor e, normalmente, não
precisam de alocação de heap. Os tipos de estrutura não dão suporte à herança especificada pelo usuário, e todos
os tipos de structs são herdados implicitamente do tipo object .
Um tipo interface define um contrato como um conjunto nomeado de membros da função pública. Um class
ou struct que implementa um interface deve fornecer implementações de membros da função da interface.
Um interface pode herdar de várias interfaces base e um class ou struct pode implementar várias interfaces.
Um tipo delegate representa referências aos métodos com uma lista de parâmetros e tipo de retorno específicos.
Delegados possibilitam o tratamento de métodos como entidades que podem ser atribuídos a variáveis e
passadas como parâmetros. Os delegados são análogos aos tipos de função fornecidos pelas linguagens
funcionais. Eles também são parecidos com o conceito de ponteiros de função em outras linguagens, mas, ao
contrário dos ponteiros de função, os delegados são orientados a objetos e fortemente tipados.
Os tipos class , struct , interface e delegate dão suporte a genéricos e podem ser parametrizados com
outros tipos.
Um tipo enum é um tipo distinto com constantes nomeadas. Cada tipo enum tem um tipo subjacente, que deve
ser um dos oito tipos integrais. O conjunto de valores de um tipo enum é o mesmo que o conjunto de valores do
tipo subjacente.
O C# dá suporte a matrizes uni e multidimensionais de qualquer tipo. Ao contrário dos tipos listados acima, os
tipos de matriz não precisam ser declarados antes de serem usados. Em vez disso, os tipos de matriz são
construídos seguindo um nome de tipo entre colchetes. Por exemplo, int[] é uma matriz unidimensional de
int , int[,] é uma matriz bidimensional de int , e int[][] é uma matriz unidimensional da matriz
unidimensional de int .
Os tipos de valor anulável também não precisam ser declarados antes de serem usados. Para cada tipo de valor
não nulo T há um tipo de valor anulável correspondente T? , que pode conter um valor adicional, null . Por
exemplo, int? é um tipo que pode conter qualquer número inteiro de 32 bits ou o valor null .
O sistema de tipos do C# é unificado, de modo que um valor de qualquer tipo pode ser tratado como um object .
Cada tipo no C#, direta ou indiretamente, deriva do tipo de classe object , e object é a classe base definitiva de
todos os tipos. Os valores de tipos de referência são tratados como objetos simplesmente exibindo os valores
como tipo object . Os valores de tipos de valor são tratados como objetos, executando conversão boxing e
operações de conversão unboxing. No exemplo a seguir, um valor int é convertido em object e volta
novamente ao int .
using System;
class BoxingExample
{
static void Main()
{
int i = 123;
object o = i; // Boxing
int j = (int)o; // Unboxing
}
}
Quando um valor de um tipo de valor é convertido para o tipo object , uma instância object , também chamada
de "caixa", é alocada para armazenar o valor e o valor é copiado na caixa. Por outro lado, quando uma referência
object é convertida em um tipo de valor, é verificado se o object referenciado é uma caixa do tipo de valor
correto e, se a verificação for bem-sucedida, o valor na caixa será copiado.
O sistema de tipo unificado do C# significa que os tipos de valor podem se tornar objetos “sob demanda”. Devido
à unificação, as bibliotecas de finalidade geral que usam o tipo object podem ser usadas com os tipos de
referência e os tipos de valor.
Existem vários tipos de variáveis no C#, incluindo campos, elementos de matriz, variáveis locais e parâmetros. As
variáveis representam os locais de armazenamento e cada variável tem um tipo que determina quais valores
podem ser armazenados na variável, conforme mostrado abaixo.
Tipo de valor não nulo
Um valor de tipo exato
Tipos de valor anulável
Um valor null ou um valor do tipo exato
objeto
Uma referência null , uma referência a um objeto de qualquer tipo de referência ou uma referência a
um valor de qualquer tipo de valor demarcado
Tipo de classe
Uma referência null , uma referência a uma instância desse tipo de classe ou uma referência a uma
instância de uma classe derivada desse tipo de classe
Tipo de interface
Uma referência null , uma referência a uma instância de um tipo de classe que implementa esse tipo
de interface ou uma referência a um valor demarcado de um tipo de valor que implementa esse tipo de
interface
Tipo de matriz
Uma referência null , uma referência a uma instância desse tipo de matriz ou uma referência a uma
instância de um tipo de matriz compatível
Tipo delegado
Uma referência null ou uma referência a uma instância de um tipo de delegado compatível
A N T E R IO R P R Ó X IM O
{1>Expressões<1}
23/11/2019 • 3 minutes to read • Edit Online
Expressões são construídas a partir de operandos e operadores. Os operadores de uma expressão indicam quais
operações devem ser aplicadas aos operandos. Exemplos de operadores incluem + , - , * , / e new . Exemplos
de operandos incluem literais, campos, variáveis locais e expressões.
Quando uma expressão contiver vários operadores, a precedência dos operadores controla a ordem na qual os
operadores individuais são avaliados. Por exemplo, a expressão x + y * z é avaliada como x + (y * z) porque
o operador * tem precedência maior do que o operador + .
Quando ocorre um operando entre dois operadores com a mesma precedência, a associatividade dos operadores
controla a ordem na qual as operações são executadas:
Exceto para os operadores de atribuição e de União nulo, todos os operadores binários são associativos à
esquerda, o que significa que as operações são executadas da esquerda para a direita. Por exemplo, x + y + z
é avaliado como (x + y) + z .
Os operadores de atribuição, a ?? de União nula e os operadores de ??= e o operador condicional ?: são
associativos à direita, o que significa que as operações são executadas da direita para a esquerda. Por exemplo,
x = y = z é avaliado como x = (y = z) .
Precedência e associatividade podem ser controladas usando parênteses. Por exemplo, x + y * z primeiro
multiplica y por z e, em seguida, adiciona o resultado a x , mas (x + y) * z primeiro adiciona x e y e, em
seguida, multiplica o resultado por z .
A maioria dos operadores pode ser sobrecarregada. A sobrecarga de operador permite que implementações de
operador definidas pelo usuário sejam especificadas para operações em que um ou ambos os operandos são de
um tipo struct ou de classe definida pelo usuário.
C# fornece uma série de operadores para realizar operações aritméticas, lógicas, bit a bit e shift e comparações de
igualdade e ordem.
Para obter a lista completa de operadores do C# ordenada pelo nível de precedência, confira Operadores do C#.
A N T E R IO R P R Ó X IM O
Instruções
23/10/2019 • 5 minutes to read • Edit Online
As ações de um programa são expressas usando instruções. O C# oferece suporte a vários tipos diferentes de
instruções, algumas delas definidas em termos de instruções inseridas.
Um bloco permite a produção de várias instruções em contextos nos quais uma única instrução é permitida. Um
bloco é composto por uma lista de instruções escritas entre os delimitadores { e } .
Instruções de declaração são usadas para declarar constantes e variáveis locais.
Instruções de expressão são usadas para avaliar expressões. As expressões que podem ser usadas como
instruções incluem chamadas de método, alocações de objeto usando o operador new , atribuições usando = e
os operadores de atribuição compostos, operações de incremento e decremento usando os operadores ++ e --
e as expressões await .
Instruções de seleção são usadas para selecionar uma dentre várias instruções possíveis para execução com base
no valor de alguma expressão. Neste grupo estão as instruções if e switch .
Instruções de iteração são usadas para executar repetidamente uma instrução inserida. Neste grupo estão as
instruções while , do , for e foreach .
Instruções de salto são usadas para transferir o controle. Neste grupo estão as instruções break , continue , goto ,
throw , return e yield .
A instrução try ... catch é usada para capturar exceções que ocorrem durante a execução de um bloco, e a
instrução try ... finally é usada para especificar o código de finalização que é executado sempre, se uma
exceção ocorrer ou não.
As instruções checked e unchecked são usadas para controlar o contexto de verificação de estouro para
operações e conversões aritméticas do tipo integral.
A instrução lock é usada para obter o bloqueio de exclusão mútua para um determinado objeto, executar uma
instrução e, em seguida, liberar o bloqueio.
A instrução using é usada para obter um recurso, executar uma instrução e, em seguida, descartar esse recurso.
A lista a seguir contém os tipos de instruções que podem ser usados, e fornece um exemplo para cada um.
Declaração de variável local:
Instrução de expressão:
Instrução if :
Instrução switch :
Instrução while :
static void WhileStatement(string[] args)
{
int i = 0;
while (i < args.Length)
{
Console.WriteLine(args[i]);
i++;
}
}
Instrução do :
Instrução for :
Instrução foreach :
Instrução break :
Instrução continue :
static void ContinueStatement(string[] args)
{
for (int i = 0; i < args.Length; i++)
{
if (args[i].StartsWith("/"))
continue;
Console.WriteLine(args[i]);
}
}
Instrução goto :
Instrução return :
Instrução yield :
Instrução lock :
class Account
{
decimal balance;
private readonly object sync = new object();
public void Withdraw(decimal amount)
{
lock (sync)
{
if (amount > balance)
{
throw new Exception(
"Insufficient funds");
}
balance -= amount;
}
}
}
Instrução using :
A N T E R IO R P R Ó X IM O
Classes e objetos
23/10/2019 • 42 minutes to read • Edit Online
As classes são os tipos do C# mais fundamentais. Uma classe é uma estrutura de dados que combina ações
(métodos e outros membros da função) e estado (campos) em uma única unidade. Uma classe fornece uma
definição para instâncias da classe criadas dinamicamente, também conhecidas como objetos. As classes dão
suporte à herança e polimorfismo, mecanismos nos quais classes derivadas podem estender e especializar classes
base.
Novas classes são criadas usando declarações de classe. Uma declaração de classe inicia com um cabeçalho que
especifica os atributos e modificadores de classe, o nome da classe, a classe base (se fornecida) e as interfaces
implementadas pela classe. O cabeçalho é seguido pelo corpo da classe, que consiste em uma lista de declarações
de membro escrita entre os delimitadores { e } .
A seguir está uma declaração de uma classe simples chamada Point :
Instâncias de classes são criadas usando o operador new , que aloca memória para uma nova instância, chama
um construtor para inicializar a instância e retorna uma referência à instância. As instruções a seguir criam dois
objetos Point e armazenam referências a esses objetos em duas variáveis:
A memória ocupada por um objeto é recuperada automaticamente quando o objeto não está mais acessível. Não
é necessário nem possível desalocar explicitamente os objetos em C#.
Membros
Os membros de uma classe são membros estáticos ou membros de instância. Os membros estáticos pertencem
às classes e os membros de instância pertencem aos objetos (instâncias de classes).
O exemplo a seguir fornece uma visão geral dos tipos de membros que uma classe pode conter.
Constantes
Valores constantes associados à classe
Campos
Variáveis de classe
Métodos
Os cálculos e as ações que podem ser executados pela classe
Propriedades
Ações associadas à leitura e à gravação de propriedades nomeadas da classe
Indexadores
Ações associadas a instâncias de indexação da classe como uma matriz
Eventos
Notificações que podem ser geradas pela classe
Operadores
Operadores de conversões e expressão com suporte da classe
Construtores
Ações necessárias para inicializar instâncias da classe ou a própria classe
Finalizadores
Ações a serem executadas antes de as instâncias da classe serem descartadas permanentemente
Tipos
Tipos aninhados declarados pela classe
Acessibilidade
Cada membro de uma classe tem uma acessibilidade associada, que controla as regiões de texto do programa
que são capazes de acessar o membro. Existem seis formas possíveis de acessibilidade. Elas são resumidas abaixo.
public
Acesso não limitado
protected
Acesso limitado a essa classe ou classes derivadas dessa classe
internal
Acesso limitado ao assembly atual (.exe, .dll, etc.)
protected internal
Acesso limitado à classe recipiente, classes derivadas da classe recipiente ou classes dentro do mesmo
assembly
private
Acesso limitado a essa classe
private protected
Acesso limitado à classe ou classes recipiente derivadas do tipo recipiente dentro do mesmo assembly
Parâmetros de tipo
Uma definição de classe pode especificar um conjunto de parâmetros de tipo seguindo o nome da classe com
colchetes angulares com uma lista de nomes de parâmetro de tipo. Em seguida, os parâmetros de tipo podem ser
usados no corpo das declarações de classe para definir os membros da classe. No exemplo a seguir, os
parâmetros de tipo de Pair são TFirst e TSecond :
Um tipo de classe que é declarado para pegar parâmetros de tipo é chamado de tipo de classe genérica. Tipos de
struct, de interface e de delegado também podem ser genéricos. Quando a classe genérica é usada, os
argumentos de tipo devem ser fornecidos para cada um dos parâmetros de tipo:
Pair<int,string> pair = new Pair<int,string> { First = 1, Second = "two" };
int i = pair.First; // TFirst is int
string s = pair.Second; // TSecond is string
Um tipo genérico com argumentos de tipo fornecidos, como Pair<int,string> acima, é chamado de tipo
construído.
Classes base
Uma declaração de classe pode especificar uma classe base, seguindo os parâmetros de nome da classe e tipo
com dois-pontos e o nome da classe base. Omitir uma especificação de classe base é o mesmo que derivar do
object de tipo. No exemplo a seguir, a classe base de Point3D é Point e a classe base de Point é object :
Uma classe herda os membros de sua classe base. A herança significa que uma classe contém implicitamente
todos os membros de sua classe base, exceto para a instância e os construtores estáticos, além dos finalizadores
da classe base. Uma classe derivada pode adicionar novos membros aos que ela herda, mas ela não pode
remover a definição de um membro herdado. No exemplo anterior, Point3D herda os campos x e y de Point
e cada instância Point3D contém três campos: x , y e z .
Existe uma conversão implícita de um tipo de classe para qualquer um de seus tipos de classe base. Portanto, uma
variável de um tipo de classe pode referenciar uma instância dessa classe ou uma instância de qualquer classe
derivada. Por exemplo, dadas as declarações de classe anteriores, uma variável do tipo Point podem referenciar
um Point ou um Point3D :
Campos
Um campo é uma variável que está associada a uma classe ou a uma instância de uma classe.
Um campo declarado com o modificador estático define um campo estático. Um campo estático identifica
exatamente um local de armazenamento. Não importa quantas instâncias de uma classe são criadas, há sempre
apenas uma cópia de um campo estático.
Um campo declarado sem o modificador estático define um campo de instância. Cada instância de uma classe
contém uma cópia separada de todos os campos de instância dessa classe.
No exemplo a seguir, cada instância da classe Color tem uma cópia separada dos campos de instância r , g e
b , mas há apenas uma cópia dos campos estáticos Black , White , Red , Green e Blue :
Conforme mostrado no exemplo anterior, os campos somente leitura podem ser declarados com um modificador
readonly . A atribuição a um campo readonly só pode ocorrer como parte da declaração do campo ou em um
construtor na mesma classe.
Métodos
Um método é um membro que implementa um cálculo ou uma ação que pode ser executada por um objeto ou
classe. Os métodos estáticos são acessados pela classe. Os métodos de instância são acessados pelas instâncias da
classe.
Os métodos podem ter uma lista de parâmetros que representam valores ou referências de variável passadas
para o método, e um tipo de retorno, que especifica o tipo do valor calculado e retornado pelo método. Um tipo
de retorno do método será void se ele não retornar um valor.
Como os tipos, os métodos também podem ter um conjunto de parâmetros de tipo, para que os quais os
argumentos de tipo devem ser especificados quando o método é chamado. Ao contrário dos tipos, os argumentos
de tipo geralmente podem ser inferidos de argumentos de uma chamada de método e não precisam ser
fornecidos explicitamente.
A assinatura de um método deve ser exclusiva na classe na qual o método é declarado. A assinatura de um
método consiste no nome do método, número de parâmetros de tipo e número, modificadores e tipos de seus
parâmetros. A assinatura de um método não inclui o tipo de retorno.
Parâmetros
Os parâmetros são usados para passar valores ou referências de variável aos métodos. Os parâmetros de um
método obtêm seus valores reais de argumentos que são especificados quando o método é invocado. Há quatro
tipos de parâmetros: parâmetros de valor, parâmetros de referência, parâmetros de saída e matrizes de
parâmetros.
Um parâmetro de valor é usado para passar argumentos de entrada. Um parâmetro de valor corresponde a uma
variável local que obtém seu valor inicial do argumento passado para o parâmetro. As modificações em um
parâmetro de valor não afetam o argumento passado para o parâmetro.
Os parâmetros de valor podem ser opcionais, especificando um valor padrão para que os argumentos
correspondentes possam ser omitidos.
Um parâmetro de referência é usado para passar argumentos por referência. O argumento passado para um
parâmetro de referência deve ser uma variável com um valor definido e, durante a execução do método, o
parâmetro de referência representa o mesmo local de armazenamento como a variável de argumento. Um
parâmetro de referência é declarado com o modificador ref . O exemplo a seguir mostra o uso de parâmetros
ref .
using System;
class RefExample
{
static void Swap(ref int x, ref int y)
{
int temp = x;
x = y;
y = temp;
}
public static void SwapExample()
{
int i = 1, j = 2;
Swap(ref i, ref j);
Console.WriteLine($"{i} {j}"); // Outputs "2 1"
}
}
Um parâmetro de saída é usado para passar argumentos por referência. Ele é semelhante a um parâmetro de
referência, exceto que ele não requer que você atribua explicitamente um valor ao argumento fornecido pelo
chamador. Um parâmetro de saída é declarado com o modificador out . O exemplo a seguir mostra o uso de
parâmetros out usando a sintaxe introduzida no C# 7.
using System;
class OutExample
{
static void Divide(int x, int y, out int result, out int remainder)
{
result = x / y;
remainder = x % y;
}
public static void OutUsage()
{
Divide(10, 3, out int res, out int rem);
Console.WriteLine("{0} {1}", res, rem); // Outputs "3 1"
}
}
}
Uma matriz de parâmetros permite que um número variável de argumentos sejam passados para um método.
Uma matriz de parâmetro é declarada com o modificador params . Somente o último parâmetro de um método
pode ser uma matriz de parâmetros e o tipo de uma matriz de parâmetros deve ser um tipo de matriz
unidimensional. Os métodos Write e WriteLine da classe System.Console são bons exemplos de uso da matriz de
parâmetros. Eles são declarados como segue.
Dentro de um método que usa uma matriz de parâmetros, a matriz de parâmetros se comporta exatamente como
um parâmetro regular de um tipo de matriz. No entanto, em uma invocação de um método com uma matriz de
parâmetros, é possível passar um único argumento do tipo da matriz de parâmetro ou qualquer número de
argumentos do tipo de elemento da matriz de parâmetros. No último caso, uma instância de matriz é
automaticamente criada e inicializada com os argumentos determinados. Esse exemplo
Console.WriteLine("x={0} y={1} z={2}", x, y, z);
using System;
class Squares
{
public static void WriteSquares()
{
int i = 0;
int j;
while (i < 10)
{
j = i * i;
Console.WriteLine($"{i} x {i} = {j}");
i = i + 1;
}
}
}
O C# requer que uma variável local seja atribuída definitivamente antes de seu valor poder ser obtido. Por
exemplo, se a declaração do i anterior não incluísse um valor inicial, o compilador relataria um erro para usos
subsequentes de i porque i não seria definitivamente atribuído a esses pontos do programa.
Um método pode usar instruções return para retornar o controle é pelo chamador. Em um método que retorna
void , as instruções return não podem especificar uma expressão. Em um método que retorna não nulo, as
instruções return devem incluir uma expressão que calcula o valor retornado.
Métodos estáticos e de instância
Um método declarado com um modificador estático é um método estático. Um método estático não funciona em
uma instância específica e pode acessar diretamente apenas membros estáticos.
Um método declarado sem um modificador estático é um método de instância. Um método de instância opera
em uma instância específica e pode acessar membros estáticos e de instância. A instância em que um método de
instância foi invocado pode ser explicitamente acessada como this . É um erro se referir a this em um método
estático.
A seguinte classe Entity tem membros estáticos e de instância.
class Entity
{
static int nextSerialNo;
int serialNo;
public Entity()
{
serialNo = nextSerialNo++;
}
public int GetSerialNo()
{
return serialNo;
}
public static int GetNextSerialNo()
{
return nextSerialNo;
}
public static void SetNextSerialNo(int value)
{
nextSerialNo = value;
}
}
Cada instância Entity contém um número de série (e, possivelmente, outras informações que não são
mostradas aqui). O construtor Entity (que é como um método de instância) inicializa a nova instância com o
próximo número de série disponível. Como o construtor é um membro de instância, ele tem permissão para
acessar tanto o campo de instância serialNo e o campo estático nextSerialNo .
Os métodos estáticos GetNextSerialNo e SetNextSerialNo podem acessar o campo estático nextSerialNo , mas
seria um erro para eles acessar diretamente o campo de instância serialNo .
O exemplo a seguir mostra o uso da classe Entity.
using System;
class EntityExample
{
public static void Usage()
{
Entity.SetNextSerialNo(1000);
Entity e1 = new Entity();
Entity e2 = new Entity();
Console.WriteLine(e1.GetSerialNo()); // Outputs "1000"
Console.WriteLine(e2.GetSerialNo()); // Outputs "1001"
Console.WriteLine(Entity.GetNextSerialNo()); // Outputs "1002"
}
}
Observe que os métodos estáticos SetNextSerialNo e GetNextSerialNo são invocados na classe enquanto o
método de instância GetSerialNo é chamado em instâncias da classe.
Métodos abstratos, virtuais e de substituição
Quando uma declaração de método de instância inclui um modificador virtual , o método deve ser um método
virtual. Quando nenhum modificador virtual estiver presente, o método será um método não virtual.
Quando um método virtual é invocado, o tipo de tempo de execução da instância para o qual essa invocação
ocorre determina a implementação real do método para invocar. Em uma invocação de método não virtual, o tipo
de tempo de compilação da instância é o fator determinante.
Um método virtual pode ser substituído em uma classe derivada. Quando uma declaração de método de instância
inclui um modificador de substituição, o método substitui um método virtual herdado com a mesma assinatura.
Enquanto uma declaração de método virtual apresenta um novo método, uma declaração de método de
substituição restringe um método virtual herdado existente fornecendo uma nova implementação do método.
Um método abstrato é um método virtual sem implementação. Um método abstrato é declarado com o
modificador abstrato e é permitido somente em uma classe que também é declarada como abstrata. Um método
abstrato deve ser substituído em cada classe derivada não abstrata.
O exemplo a seguir declara uma classe abstrata, Expression , que representa um nó de árvore de expressão e três
classes derivadas, Constant , VariableReference e Operation , que implementam nós de árvore de expressão para
operações aritméticas, referências de variável e constantes. (Isso é semelhante, mas não deve ser confundido com
os tipos de árvore de expressão).
using System;
using System.Collections.Generic;
public abstract class Expression
{
public abstract double Evaluate(Dictionary<string,object> vars);
}
public class Constant: Expression
{
double value;
public Constant(double value)
{
this.value = value;
}
public override double Evaluate(Dictionary<string,object> vars)
{
return value;
}
}
public class VariableReference: Expression
{
string name;
public VariableReference(string name)
{
this.name = name;
}
public override double Evaluate(Dictionary<string,object> vars)
{
object value = vars[name];
if (value == null)
{
throw new Exception("Unknown variable: " + name);
}
return Convert.ToDouble(value);
}
}
public class Operation: Expression
{
Expression left;
char op;
Expression right;
public Operation(Expression left, char op, Expression right)
{
this.left = left;
this.op = op;
this.right = right;
}
public override double Evaluate(Dictionary<string,object> vars)
{
double x = left.Evaluate(vars);
double y = right.Evaluate(vars);
switch (op) {
case '+': return x + y;
case '-': return x - y;
case '*': return x * y;
case '/': return x / y;
}
throw new Exception("Unknown operator");
}
}
As quatro classes anteriores podem ser usadas para modelar expressões aritméticas. Por exemplo, usando
instâncias dessas classes, a expressão x + 3 pode ser representada da seguinte maneira.
Expression e = new Operation(
new VariableReference("x"),
'+',
new Constant(3));
O método Evaluate de uma instância Expression é chamado para avaliar a expressão especificada e produzir
um valor double . O método recebe um argumento Dictionary que contém nomes de variáveis (como chaves
das entradas) e valores (como valores das entradas). Como Evaluate é um método abstrato, classes não
abstratas derivadas de Expression devem substituir Evaluate .
Uma implementação de Evaluate do Constant retorna apenas a constante armazenada. Uma implementação de
VariableReference consulta o nome de variável no dicionário e retorna o valor resultante. Uma implementação
de Operation primeiro avalia os operandos esquerdo e direito (chamando recursivamente seus métodos
Evaluate ) e, em seguida, executa a operação aritmética determinada.
O seguinte programa usa as classes Expression para avaliar a expressão x * (y + 2) para valores diferentes de
x e y .
using System;
using System.Collections.Generic;
class InheritanceExample
{
public static void ExampleUsage()
{
Expression e = new Operation(
new VariableReference("x"),
'*',
new Operation(
new VariableReference("y"),
'+',
new Constant(2)
)
);
Dictionary<string,object> vars = new Dictionary<string, object>();
vars["x"] = 3;
vars["y"] = 5;
Console.WriteLine(e.Evaluate(vars)); // Outputs "21"
vars["x"] = 1.5;
vars["y"] = 9;
Console.WriteLine(e.Evaluate(vars)); // Outputs "16.5"
}
}
Sobrecarga de método
A sobrecarga de método permite que vários métodos na mesma classe tenham o mesmo nome, contanto que
tenham assinaturas exclusivas. Ao compilar uma invocação de um método sobrecarregado, o compilador usa a
resolução de sobrecarga para determinar o método específico para invocar. A resolução de sobrecarga localizará
o método que melhor corresponde aos argumentos ou relatará um erro se nenhuma correspondência for
encontrada. O exemplo a seguir mostra a resolução de sobrecarga em vigor. O comentário para cada invocação
no método UsageExample mostra qual método é realmente chamado.
using System;
class OverloadingExample
{
static void F()
{
Console.WriteLine("F()");
}
static void F(object x)
{
Console.WriteLine("F(object)");
}
static void F(int x)
{
Console.WriteLine("F(int)");
}
static void F(double x)
{
Console.WriteLine("F(double)");
}
static void F<T>(T x)
{
Console.WriteLine("F<T>(T)");
}
static void F(double x, double y)
{
Console.WriteLine("F(double, double)");
}
public static void UsageExample()
{
F(); // Invokes F()
F(1); // Invokes F(int)
F(1.0); // Invokes F(double)
F("abc"); // Invokes F<string>(string)
F((double)1); // Invokes F(double)
F((object)1); // Invokes F(object)
F<int>(1); // Invokes F<int>(int)
F(1, 1); // Invokes F(double, double)
}
}
Conforme mostrado no exemplo, um determinado método sempre pode ser selecionado ao converter
explicitamente os argumentos para os tipos de parâmetro exatos e/ou fornecendo explicitamente os argumentos
de tipo.
NOTE
Este exemplo cria uma classe MyList , que não é igual ao .NET padrão System.Collections.Generic.List<T>. Ele ilustra os
conceitos necessários para esse tour, mas não serve como substituto para essa classe.
// Fields
T[] items;
int count;
// Constructor
public MyList(int capacity = defaultCapacity)
{
items = new T[capacity];
}
// Properties
public int Count => count;
// Indexer
public T this[int index]
{
get
{
return items[index];
}
set
{
items[index] = value;
OnChanged();
}
}
// Methods
public void Add(T item)
{
if (count == Capacity) Capacity = count * 2;
items[count] = item;
count++;
OnChanged();
}
protected virtual void OnChanged() =>
Changed?.Invoke(this, EventArgs.Empty);
// Event
public event EventHandler Changed;
// Operators
public static bool operator ==(MyList<T> a, MyList<T> b) =>
Equals(a, b);
Construtores
O C# dá suporte aos construtores estáticos e de instância. Um construtor de instância é um membro que
implementa as ações necessárias para inicializar uma instância de uma classe. Um construtor estático é um
membro que implementa as ações necessárias para inicializar uma classe quando ele for carregado pela primeira
vez.
Um construtor é declarado como um método sem nenhum tipo de retorno e o mesmo nome que a classe
continente. Se uma declaração de construtor inclui um modificador estático, ela declara um construtor estático.
Caso contrário, ela declara um construtor de instância.
Construtores de instância podem ser sobrecarregados e ter parâmetros opcionais. Por exemplo, a classe
MyList<T> declara um construtor de instância com um único parâmetro int opcional. Os construtores de
instância são invocados usando o operador new . As seguintes instruções alocam duas instâncias MyList<string>
usando o construtor da classe MyList com e sem o argumento opcional.
Diferentemente de outros membros, construtores de instância não são herdados e uma classe não tem nenhum
construtor de instância que não os que são realmente declarados na classe. Se nenhum construtor de instância for
fornecido para uma classe, então um construtor vazio sem parâmetros será fornecido automaticamente.
Propriedades
As propriedades são uma extensão natural dos campos. Elas são denominadas membros com tipos associados, e
a sintaxe para acessar os campos e as propriedades é a mesma. No entanto, diferentemente dos campos, as
propriedades não denotam locais de armazenamento. Em vez disso, as propriedades têm acessadores que
especificam as instruções a serem executadas quando os valores forem lidos ou gravados.
Uma propriedade é declarada como um campo, exceto quando a declaração termina com um acessador get e/ou
um acessador set gravado entre os delimitadores { e } em vez de terminar com um ponto-e-vírgula. Uma
propriedade que tem um acessador get e um acessador set é uma propriedade de leitura -gravação. Uma
propriedade que tem apenas um acessador get é uma propriedade somente leitura, e uma propriedade que tem
apenas um acessador set é uma propriedade somente gravação.
Um acessador get corresponde a um método sem parâmetros com um valor retornado do tipo de propriedade.
Exceto como o destino de uma atribuição, quando uma propriedade é referenciada em uma expressão, o
acessador get da propriedade é invocado para calcular o valor da propriedade.
Um acessador set corresponde a um método com um parâmetro único chamado valor e nenhum tipo de retorno.
Quando uma propriedade é referenciada como o destino de uma atribuição ou como o operando do + + ou --, o
acessador set é invocado com um argumento que fornece o novo valor.
A classe MyList<T> declara duas propriedades, Count e Capacity , que são somente leitura e leitura/gravação,
respectivamente. Veja a seguir um exemplo de uso dessas propriedades:
Os indexadores podem ser sobrecarregados, o que significa que uma classe pode declarar vários indexadores,
desde que o número ou os tipos de seus parâmetros sejam diferentes.
Eventos
Um evento é um membro que permite que uma classe ou objeto forneça notificações. Um evento é declarado
como um campo exceto se a declaração incluir uma palavra-chave do evento e o tipo deverá ser um tipo
delegado.
Em uma classe que declara um membro de evento, o evento se comporta exatamente como um campo de um
tipo delegado (desde que o evento não seja abstrato e não declare acessadores). O campo armazena uma
referência a um delegado que representa os manipuladores de eventos que foram adicionados ao evento. Se
nenhum manipulador de evento estiver presente, o campo será null .
A classe MyList<T> declara um membro único de evento chamado Changed , que indica que um novo item foi
adicionado à lista. O evento Alterado é gerado pelo método virtual OnChanged , que primeiro verifica se o evento é
null (o que significa que nenhum manipulador está presente). A noção de gerar um evento é precisamente
equivalente a invocar o delegado representado pelo evento — assim, não há constructos de linguagem especial
para gerar eventos.
Os clientes reagem a eventos por meio de manipuladores de eventos. Os manipuladores de eventos são
conectados usando o operador += e removidos usando o operador -= . O exemplo a seguir anexa um
manipulador de eventos para o evento Changed de um MyList<string> .
class EventExample
{
static int changeCount;
static void ListChanged(object sender, EventArgs e)
{
changeCount++;
}
public static void Usage()
{
MyList<string> names = new MyList<string>();
names.Changed += new EventHandler(ListChanged);
names.Add("Liz");
names.Add("Martha");
names.Add("Beth");
Console.WriteLine(changeCount); // Outputs "3"
}
}
Para cenários avançados nos quais o controle do armazenamento subjacente de um evento é desejado, uma
declaração de evento pode fornecer explicitamente acessadores add e remove , que são um pouco semelhantes
ao acessador set de uma propriedade.
Operadores
Um operador é um membro que define o significado da aplicação de um operador de expressão específico para
instâncias de uma classe. Três tipos de operadores podem ser definidos: operadores unários, operadores binários
e operadores de conversão. Todos os operadores devem ser declarados como public e static .
A classe MyList<T> declara dois operadores, operator == e operator != e, portanto, dá um novo significado
para as expressões que aplicam esses operadores a instâncias MyList . Especificamente, os operadores definem a
igualdade de duas instâncias MyList<T> ao comparar cada um dos objetos contidos usando os métodos Equals.
O exemplo a seguir usa o operador == para comparar duas instâncias MyList<int> .
O primeiro Console.WriteLine gera True porque as duas listas contêm o mesmo número de objetos com os
mesmos valores na mesma ordem. Como MyList<T> não definiu operator == , o primeiro Console.WriteLine
geraria False porque a e b referenciam diferentes instâncias MyList<int> .
Finalizadores
Um finalizador é um membro que implementa as ações necessárias para finalizar uma instância de uma classe.
Os finalizadores não podem ter parâmetros, eles não podem ter modificadores de acessibilidade e não podem ser
chamados explicitamente. O finalizador de uma instância é invocado automaticamente durante a coleta de lixo.
O coletor de lixo tem latitude ampla ao decidir quando coletar objetos e executar os finalizadores.
Especificamente, o tempo de invocações de finalizador não é determinístico e finalizadores podem ser executados
em qualquer thread. Para esses e outros motivos, as classes devem implementar os finalizadores apenas quando
não houver outras soluções viáveis.
A instrução using fornece uma abordagem melhor para a destruição de objetos.
A N T E R IO R P R Ó X IM O
Structs
23/10/2019 • 4 minutes to read • Edit Online
Como classes, os structs são estruturas de dados que podem conter membros de dados e os membros da função,
mas, ao contrário das classes, as estruturas são tipos de valor e não precisam de alocação de heap. Uma variável
de um tipo struct armazena diretamente os dados do struct, enquanto que uma variável de um tipo de classe
armazena uma referência a um objeto alocado dinamicamente. Os tipos de struct não dão suporte à herança
especificada pelo usuário, e todos os tipos de structs são herdados implicitamente do tipo ValueType, que, por sua
vez, é herdado implicitamente de object .
Os structs são particularmente úteis para estruturas de dados pequenas que têm semântica de valor. Números
complexos, pontos em um sistema de coordenadas ou pares chave-valor em um dicionário são exemplos de
structs. O uso de structs, em vez de classes para estruturas de dados pequenas, pode fazer uma grande diferença
no número de alocações de memória que um aplicativo executa. Por exemplo, o programa a seguir cria e inicializa
uma matriz de 100 pontos. Com Point implementado como uma classe, 101 objetos separados são instanciados
— um para a matriz e um para os elementos de 100.
struct Point
{
public int x, y;
public Point(int x, int y)
{
this.x = x;
this.y = y;
}
}
Agora, somente um objeto é instanciado — um para a matriz — e as instâncias Point são armazenadas em linha
na matriz.
Os construtores struct são invocados com o operador new , semelhante a um construtor de classe. Porém, em vez
de alocar dinamicamente um objeto no heap gerenciado e retornar uma referência a ele, um construtor de struct
simplesmente retorna o valor do struct (normalmente em um local temporário na pilha), e esse valor é, então,
copiado conforme a necessidade.
Com classes, é possível que duas variáveis referenciem o mesmo objeto e, portanto, é possível que operações em
uma variável afetem o objeto referenciado por outra variável. Com structs, as variáveis têm sua própria cópia dos
dados e não é possível que as operações em um afetem o outro. Por exemplo, a saída produzida pelo seguinte
fragmento de código depende de o ponto ser uma classe ou um struct.
Point a = new Point(10, 10);
Point b = a;
a.x = 20;
Console.WriteLine(b.x);
Se Point for uma classe, a saída será 20 porque a e b referenciam o mesmo objeto. Se Point for um struct, a
saída será 10 porque a atribuição de a para b cria uma cópia do valor e essa cópia não é afetada pela atribuição
seguinte para a.x .
O exemplo anterior destaca duas das limitações dos structs. Primeiro, copiar um struct inteiro é, geralmente,
menos eficiente do que copiar uma referência de objeto, então a passagem de atribuição e de valor do parâmetro
pode ser mais custosa com structs que com tipos de referência. Segundo, com exceção dos parâmetros in , ref
e out , não é possível criar referências para structs, o que rege o uso em diversas situações.
A N T E R IO R P R Ó X IM O
Matrizes
31/10/2019 • 5 minutes to read • Edit Online
Uma matriz é uma estrutura de dados que contém algumas variáveis acessadas por meio de índices calculados.
As variáveis contidas em uma matriz, também chamadas de elementos da matriz, são todas do mesmo tipo, e
esse tipo é chamado de tipo de elemento da matriz.
Os tipos de matriz são tipos de referência, e a declaração de uma variável de matriz simplesmente reserva espaço
para uma referência a uma instância de matriz. As instâncias reais da matriz são criadas dinamicamente em tempo
de execução usando o operador new. A operação new especifica a duração da nova instância de matriz, que
depois fica fixa para o tempo de vida da instância. Os índices dos elementos de uma matriz variam de 0 a
Length - 1 . O operador new inicializa automaticamente os elementos de uma matriz usando o valor padrão,
que, por exemplo, é zero para todos os tipos numéricos e null para todos os tipos de referência.
O exemplo a seguir cria uma matriz de elementos int , inicializa a matriz e imprime o conteúdo da matriz.
using System;
class ArrayExample
{
static void Main()
{
int[] a = new int[10];
for (int i = 0; i < a.Length; i++)
{
a[i] = i * i;
}
for (int i = 0; i < a.Length; i++)
{
Console.WriteLine($"a[{i}] = {a[i]}");
}
}
}
Este exemplo cria e opera em uma matriz unidimensional. O C# também oferece suporte a matrizes
multidimensionais. O número de dimensões de um tipo de matriz, também conhecido como classificação do
tipo de matriz, é o número um mais o número de vírgulas escrito entre os colchetes do tipo de matriz. O exemplo
a seguir aloca uma matriz unidimensional, bidimensional e tridimensional, respectivamente.
A matriz a1 contém 10 elementos, a matriz a2 contém 50 (10 × 5) elementos e a matriz a3 contém 100 (10 ×
5 × 2) elementos. O tipo do elemento de uma matriz pode ser qualquer tipo, incluindo um tipo de matriz. Uma
matriz com elementos de um tipo de matriz é chamada às vezes de matriz denteada, pois os tamanhos das
matrizes do elemento nem sempre precisam ser iguais. O exemplo a seguir aloca uma matriz de matrizes de int :
A primeira linha cria uma matriz com três elementos, cada um do tipo int[] , e cada um com um valor inicial de
null . As linhas subsequentes inicializam os três elementos com referências às instâncias individuais da matriz de
tamanhos variados.
O operador new permite que os valores iniciais dos elementos da matriz sejam especificados com um
inicializador de matriz, que é uma lista de expressões escritas entre os delimitadores { e } . O exemplo a
seguir aloca e inicializa um int[] com três elementos.
Observe que o tamanho da matriz é inferido do número de expressões entre { e }. A variável local e declarações de
campo podem ser reduzidas ainda mais, de modo que o tipo de matriz não precise ser redefinido.
A N T E R IO R P R Ó X IM O
Interfaces
23/10/2019 • 3 minutes to read • Edit Online
Uma interface define um contrato que pode ser implementado por classes e estruturas. Uma interface pode
conter métodos, propriedades, eventos e indexadores. Uma interface não fornece implementações dos membros
que define — ela simplesmente especifica os membros que devem ser fornecidos por classes ou estruturas que
implementam a interface.
As interfaces podem empregar a herança múltipla. No exemplo a seguir, a interface IComboBox herda de
ITextBox e IListBox .
interface IControl
{
void Paint();
}
interface ITextBox: IControl
{
void SetText(string text);
}
interface IListBox: IControl
{
void SetItems(string[] items);
}
interface IComboBox: ITextBox, IListBox {}
Classes e structs podem implementar várias interfaces. No exemplo a seguir, a classe EditBox implementa
IControl e IDataBound .
interface IDataBound
{
void Bind(Binder b);
}
public class EditBox: IControl, IDataBound
{
public void Paint() { }
public void Bind(Binder b) { }
}
Quando uma classe ou struct implementa uma interface específica, as instâncias dessa classe ou struct podem ser
convertidas implicitamente para esse tipo de interface. Por exemplo
Em casos nos quais uma instância não é conhecida por ser estática para implementar uma interface específica, é
possível usar conversões de tipo dinâmico. Por exemplo, as instruções a seguir usam conversões de tipo dinâmico
para obter as implementações de interface de IControl e IDataBound do objeto. Como o tipo de tempo de
execução real do objeto é EditBox , as conversões são bem-sucedidas.
object obj = new EditBox();
IControl control = (IControl)obj;
IDataBound dataBound = (IDataBound)obj;
Na classe EditBox anterior, o método Paint da interface IControl e o método Bind da interface IDataBound
são implementados usando membros públicos. C# também oferece suporte explícito a implementações de
membros de interface, permitindo que a classe ou struct evite tornar os membros públicos. Uma implementação
de membro de interface explícita é escrita usando o nome do membro de interface totalmente qualificado. Por
exemplo, a classe EditBox pode implementar os métodos IControl.Paint e IDataBound.Bind usando
implementações de membros de interface explícita da seguinte maneira.
Os membros de interface explícita só podem ser acessados por meio do tipo de interface. Por exemplo, a
implementação de IControl.Paint fornecida pela classe EditBox anterior só pode ser chamada convertendo
primeiro a referência EditBox ao tipo de interface IControl .
A N T E R IO R P R Ó X IM O
Enums
23/10/2019 • 3 minutes to read • Edit Online
Um tipo enum é um tipo de valor diferente com um conjunto de constantes nomeadas. Você define enums
quando precisa definir um tipo que pode ter um conjunto de valores discretos. Eles usam um dos tipos de valor
integral como o armazenamento subjacente. Eles fornecem um significado semântico aos valores discretos.
O exemplo a seguir declara e usa um tipo enum chamado Color com três valores de constantes, Red , Green e
Blue .
using System;
enum Color
{
Red,
Green,
Blue
}
class EnumExample
{
static void PrintColor(Color color)
{
switch (color)
{
case Color.Red:
Console.WriteLine("Red");
break;
case Color.Green:
Console.WriteLine("Green");
break;
case Color.Blue:
Console.WriteLine("Blue");
break;
default:
Console.WriteLine("Unknown color");
break;
}
}
static void Main()
{
Color c = Color.Red;
PrintColor(c);
PrintColor(Color.Blue);
}
}
Cada tipo enum tem um tipo integral correspondente chamado de tipo subjacente do tipo enum . Um tipo enum
que não declara explicitamente um tipo subjacente tem um tipo subjacente de int . Um formato de
armazenamento do tipo enum e o intervalo de valores possíveis são determinados pelo seu tipo subjacente. O
conjunto de valores que um tipo enum pode lidar não é limitado pelos seus membros enum . Em particular,
qualquer valor do tipo subjacente de um enum pode ser convertido em um tipo enum e é um valor válido
diferente do que o do tipo enum .
O exemplo a seguir declara um tipo enum chamado Alignment com um tipo subjacente de sbyte .
enum Alignment: sbyte
{
Left = -1,
Center = 0,
Right = 1
}
Conforme mostrado no exemplo anterior, uma declaração de membro enum pode incluir uma expressão
constante que especifica o valor do membro. O valor da constante para cada membro enum deve estar no
intervalo do tipo subjacente do enum . Quando uma declaração de membro enum não especifica explicitamente
um valor, o membro recebe o valor zero (se ele é o primeiro membro no tipo enum ) ou o valor do membro enum
textualmente precedente mais um.
Os valores Enum podem ser convertidos em valores integrais e vice-versa usando conversões de tipo. Por
exemplo:
O valor padrão de qualquer tipo enum é o valor integral zero convertido para o tipo enum . Em casos nos quais as
variáveis são inicializadas automaticamente como um valor padrão, esse é o valor fornecido para variáveis de
tipos enum . Para que o valor padrão de um tipo enum fique facilmente disponível, o 0 literal é convertido
implicitamente para qualquer tipo enum . Dessa forma, o seguinte é permitido.
Color c = 0;
A N T E R IO R P R Ó X IM O
Delegados
23/10/2019 • 3 minutes to read • Edit Online
Um delegado é um tipo que representa referências aos métodos com uma lista de parâmetros e tipo de retorno
específicos. Delegados possibilitam o tratamento de métodos como entidades que podem ser atribuídos a
variáveis e passadas como parâmetros. Os delegados são parecidos com o conceito de ponteiros de função em
outras linguagens, mas ao contrário dos ponteiros de função, os delegados são orientados a objetos e fortemente
tipados.
O exemplo a seguir declara e usa um tipo delegado chamado Function .
using System;
delegate double Function(double x);
class Multiplier
{
double factor;
public Multiplier(double factor)
{
this.factor = factor;
}
public double Multiply(double x)
{
return x * factor;
}
}
class DelegateExample
{
static double Square(double x)
{
return x * x;
}
static double[] Apply(double[] a, Function f)
{
double[] result = new double[a.Length];
for (int i = 0; i < a.Length; i++) result[i] = f(a[i]);
return result;
}
static void Main()
{
double[] a = {0.0, 0.5, 1.0};
double[] squares = Apply(a, Square);
double[] sines = Apply(a, Math.Sin);
Multiplier m = new Multiplier(2.0);
double[] doubles = Apply(a, m.Multiply);
}
}
Uma instância do tipo delegado Function pode fazer referência a qualquer método que usa um argumento
double e retorna um valor double . O método Apply aplica uma determinada função aos elementos de um
double[] , retornando um double[] com os resultados. No método Main , Apply é usado para aplicar três
funções diferentes para um double[] .
Um delegado pode referenciar um método estático (como Square ou Math.Sin no exemplo anterior) ou um
método de instância (como m.Multiply no exemplo anterior). Um delegado que referencia um método de
instância também referencia um objeto específico, e quando o método de instância é invocado por meio do
delegado, esse objeto se torna this na invocação.
Os delegados podem ser criados usando funções anônimas, que são "métodos embutidos" criados
dinamicamente. As funções anônimas podem ver as variáveis locais dos métodos ao redor. Assim, o exemplo de
multiplicador acima pode ser gravado mais facilmente sem usar uma classe de multiplicador:
Uma propriedade interessante e útil de um delegado é que ele não sabe ou se importa com a classe do método
que referencia; o que importa é que o método referenciado tem os mesmos parâmetros e o tipo de retorno do
delegado.
A N T E R IO R P R Ó X IM O
Atributos
31/10/2019 • 3 minutes to read • Edit Online
Tipos, membros e outras entidades em um programa C# dão suporte a modificadores que controlam
determinados aspectos de seu comportamento. Por exemplo, a acessibilidade de um método é controlada usando
os modificadores public , protected , internal e private . O C# generaliza essa funcionalidade, de modo que os
tipos definidos pelo usuário de informações declarativas podem ser anexados a entidades de programa e
recuperados no tempo de execução. Os programas especificam essas informações declarativas adicionais,
definindo e usando os atributos.
O exemplo a seguir declara um atributo HelpAttribute que pode ser colocado em entidades de programa para
fornecem links para a documentação associada.
using System;
Todas as classes de atributo derivam da classe base Attribute fornecida pela biblioteca padrão. Os atributos
podem ser aplicados, fornecendo seu nome, junto com quaisquer argumentos, dentro dos colchetes pouco antes
da declaração associada. Se o nome de um atributo termina em Attribute , essa parte do nome pode ser omitida
quando o atributo é referenciado. Por exemplo, o atributo HelpAttribute pode ser usado da seguinte maneira.
[Help("https://docs.microsoft.com/dotnet/csharp/tour-of-csharp/attributes")]
public class Widget
{
[Help("https://docs.microsoft.com/dotnet/csharp/tour-of-csharp/attributes",
Topic = "Display")]
public void Display(string text) {}
}
Este exemplo anexa um HelpAttribute à classe Widget . Ele adiciona outro HelpAttribute ao método Display na
classe. Os construtores públicos de uma classe de atributo controlam as informações que devem ser fornecidas
quando o atributo é anexado a uma entidade de programa. As informações adicionais podem ser fornecidas ao
referenciar propriedades públicas de leitura-gravação da classe de atributo (como a referência anterior à
propriedade Topic ).
Os metadados definidos por atributos podem ser lidos e manipulados em tempo de execução usando reflexão.
Quando um atributo específico for solicitado usando esta técnica, o construtor para a classe de atributo será
invocado com as informações fornecidas na origem do programa e a instância do atributo resultante será
retornada. Se forem fornecidas informações adicionais por meio de propriedades, essas propriedades serão
definidas para os valores fornecidos antes que a instância do atributo seja retornada.
O exemplo de código a seguir demonstra como obter as instâncias HelpAttribute associadas à classe Widget e
seu método Display .
if (widgetClassAttributes.Length > 0)
{
HelpAttribute attr = (HelpAttribute)widgetClassAttributes[0];
Console.WriteLine($"Widget class help URL : {attr.Url} - Related topic : {attr.Topic}");
}
if (displayMethodAttributes.Length > 0)
{
HelpAttribute attr = (HelpAttribute)displayMethodAttributes[0];
Console.WriteLine($"Display method help URL : {attr.Url} - Related topic : {attr.Topic}");
}
Console.ReadLine();
A N T E R IO R
Novidades no C# 8.0
27/11/2019 • 28 minutes to read • Edit Online
Assim como a maioria das structs, o método ToString() não modifica o estado. É possível indicar isso
adicionando o modificador readonly à declaração de ToString() :
A alteração anterior gera um aviso do compilador, porque ToString acessa a propriedade Distance , que não está
marcada readonly :
warning CS8656: Call to non-readonly member 'Point.Distance.get' from a 'readonly' member results in an
implicit copy of 'this'
O compilador avisa quando há a necessidade de criar uma cópia de defesa. A propriedade Distance não altera o
estado, portanto você pode corrigir esse aviso adicionando o modificador readonly à declaração:
Observe que o modificador de readonly é necessário em uma propriedade somente leitura. O compilador não
pressupõe que get acessadores não modifiquem o estado; Você deve declarar readonly explicitamente. As
propriedades implementadas automaticamente são uma exceção; o compilador tratará todos os getters
autoimplementados como ReadOnly, portanto, aqui não há necessidade de adicionar o modificador readonly às
propriedades X e Y .
O compilador impõe a regra que readonly Membros não modificam o estado. O método a seguir não será
compilado, a menos que você remova o modificador de readonly :
Esse recurso permite que você especifique sua intenção de design para que o compilador possa impô-la e faça
otimizações com base nessa intenção. Você pode saber mais sobre membros somente leitura no artigo de
referência de linguagem em readonly .
Se seu aplicativo definiu um tipo RGBColor que é construído a partir dos componentes R , G e B , você poderia
converter um valor Rainbow em seus valores RGB usando o método a seguir que contém uma expressão de
opção:
Padrões da propriedade
O padrão da propriedade permite que você compare as propriedades do objeto examinado. Considere um site
de comércio eletrônico que deve calcular o imposto da venda com base no endereço do comprador. Essa
computação não é uma responsabilidade principal de uma classe de Address . Ele mudará ao longo do tempo,
provavelmente com mais frequência do que as alterações de formato de endereço. O valor do imposto depende da
propriedade State do endereço. O método a seguir usa o padrão de propriedade para calcular o imposto da
venda de acordo com o endereço e o preço:
A correspondência de padrão cria uma sintaxe concisa para expressar esse algoritmo.
Padrões de tupla
Alguns algoritmos dependem de várias entradas. Padrões de tupla permitem que você alterne com base em
vários valores, expressadas como uma tupla. O código a seguir mostra uma expressão de comutador para o jogo
pedra, papel, tesoura:
public static string RockPaperScissors(string first, string second)
=> (first, second) switch
{
("rock", "paper") => "rock is covered by paper. Paper wins.",
("rock", "scissors") => "rock breaks scissors. Rock wins.",
("paper", "rock") => "paper covers rock. Paper wins.",
("paper", "scissors") => "paper is cut by scissors. Scissors wins.",
("scissors", "rock") => "scissors is broken by rock. Rock wins.",
("scissors", "paper") => "scissors cuts paper. Scissors wins.",
(_, _) => "tie"
};
As mensagens indicam o vencedor. O caso de descarte representa três combinações para empates ou outras
entradas de texto.
Padrões posicionais
Alguns tipos incluem um método Deconstruct que desconstrói suas propriedades em variáveis discretas. Quando
um método Deconstruct é acessível, você pode usar padrões posicionais para inspecionar as propriedades do
objeto e usar essas propriedades para um padrão. Considere a seguinte classe Point , que inclui um método
Deconstruct para criar variáveis discretas para X e Y :
Além disso, considere a seguinte enumeração que representa várias posições de um quadrante:
O método a seguir usa o padrão posicional para extrair os valores de x e y . Em seguida, ele usa uma cláusula
when para determinar o Quadrant do ponto:
Declarações using
Uma declaração using é uma declaração de variável precedida pela palavra-chave using . Ele informa ao
compilador que a variável que está sendo declarada deve ser descartada ao final do escopo delimitador. Por
exemplo, considere o seguinte código que grava um arquivo de texto:
No exemplo anterior, o arquivo é descartado quando a chave de fechamento do método é atingida. Esse é o final
do escopo no qual file é declarado. O código anterior equivale ao código a seguir que usa as instruções using
clássicas:
No exemplo anterior, o arquivo é descartado quando a chave de fechamento associada à instrução using é
atingida.
Em ambos os casos, o compilador gera a chamada para Dispose() . O compilador gerará um erro se a expressão
na instrução using não for descartável.
int M()
{
int y;
LocalFunction();
return y;
O código a seguir contém uma função local estática. Ela pode ser estática porque não acesse as variáveis no
escopo delimitador:
int M()
{
int y = 5;
int x = 7;
return Add(x, y);
Fluxos assíncronos
A partir do C# 8.0, você pode criar e consumir fluxos de forma assíncrona. Um método que retorna um fluxo
assíncrono tem três propriedades:
1. Ele é declarado com o modificador async .
2. Ela retorna um IAsyncEnumerable<T>.
3. O método contém instruções yield return para retornar elementos sucessivos no fluxo assíncrono.
O consumo de um fluxo assíncrono exige que você adicione a palavra-chave await antes da palavra-chave
foreach quando você enumera os elementos do fluxo. A adição da palavra-chave await exige o método que
enumera o fluxo assíncrono seja declarado com o modificador async e retorne um tipo permitido para um
método async . Normalmente, isso significa retornar um Task ou Task<TResult>. Também pode ser ValueTask ou
ValueTask<TResult>. Um método pode consumir e produzir um fluxo assíncrono, o que significa que retornaria
um IAsyncEnumerable<T>. O código a seguir gera uma sequência de 0 a 19, esperando 100 ms entre a geração
de cada número:
Experimente você mesmo os fluxos assíncronos em nosso tutorial sobre como criar e consumir fluxos assíncronos.
Índices e intervalos
Índices e intervalos fornecem uma sintaxe sucinta para acessar elementos únicos ou intervalos em uma sequência.
Esse suporte a idioma depende de dois novos tipos e de dois novos operadores:
System.Index representa um índice em uma sequência.
O índice do operador end ^ , que especifica que um índice é relativo ao final da sequência.
System.Range representa um subintervalo de uma sequência.
O operador Range .. , que especifica o início e o término de um intervalo como seus operandos.
Vamos começar com as regras para índices. Considere uma matriz sequence . O índice 0 é o mesmo que
sequence[0] . O índice ^0 é o mesmo que sequence[sequence.Length] . Observe que sequence[^0] gera uma
exceção, assim como sequence[sequence.Length] faz. Para qualquer número n , o índice ^n é o mesmo que
sequence.Length - n .
Um intervalo especifica o início e o final de um intervalo. O início do intervalo é inclusivo, mas o final do intervalo
é exclusivo, o que significa que o início é incluído no intervalo, mas o final não é incluído no intervalo. O intervalo
[0..^0] representa todo o intervalo, assim como [0..sequence.Length] representa todo o intervalo.
Vamos analisar alguns exemplos. Considere a matriz a seguir, anotada com seu índice do início e do final:
O código a seguir cria um subintervalo com as palavras "quick", "brown" e "fox". Ele inclui words[1] até words[3] .
O elemento words[4] não está no intervalo.
O código a seguir cria um subintervalo com "lazy" e "dog". Ele inclui words[^2] e words[^1] .O words[^0] de
índice final não está incluído:
Não apenas as matrizes dão suporte a índices e intervalos. Você também pode usar índices e intervalos com
cadeia de caracteres, Span<T>ou ReadOnlySpan<T>. Para obter mais informações, consulte suporte de tipo para
índices e intervalos.
Você pode explorar mais sobre índices e intervalos do tutorial sobre índices e intervalos.
o tipo de Coords<int> é um tipo não gerenciado em C# 8,0 e posterior. Como para qualquer tipo não gerenciado,
você pode criar um ponteiro para uma variável desse tipo ou alocar um bloco de memória na pilha para instâncias
desse tipo:
Há dois temas principais para a versão C# 7.3. Um tema fornece recursos que permitem que o código seguro
tenha o mesmo desempenho que o código não seguro. O segundo tema fornece melhorias incrementais aos
recursos existentes. Além disso, novas opções do compilador foram adicionadas nesta versão.
Os novos recursos a seguir são compatíveis com o tema de melhor desempenho para código seguro:
Você pode acessar campos fixos sem fixação.
Você pode reatribuir variáveis locais ref .
Você pode usar inicializadores em matrizes stackalloc .
Você pode usar instruções fixed com qualquer tipo compatível com um padrão.
Você pode usar restrições genéricas adicionais.
Os seguintes recursos e aprimoramentos foram feitos nos recursos existentes:
Você pode testar == e != com tipos de tupla.
Você pode usar variáveis de expressão em mais locais.
Você pode anexar atributos ao campo de suporte de propriedades autoimplementadas.
A resolução de métodos quando os argumentos se diferenciam por in foi aprimorada.
A resolução de sobrecarga agora tem menos casos ambíguos.
As novas opções do compilador são:
-publicsign para habilitar a assinatura de Software de código aberto (OSS ) de assemblies.
-pathmap para fornecer um mapeamento para diretórios de origem.
O restante deste artigo fornece detalhes e links para saber mais sobre cada uma das melhorias. Você pode
explorar esses recursos em seu ambiente usando a ferramenta global dotnet try :
1. Instale a ferramenta global dotnet-try.
2. Clone o repositório dotnet/try-samples.
3. Definir o diretório atual do subdiretório csharp7 para o repositório try-samples.
4. Execute dotnet try .
unsafe struct S
{
public fixed int myFixedField[10];
}
Em versões anteriores do C#, era preciso fixar uma variável para acessar um dos números inteiros que fazem parte
de myFixedField . Agora, o código a seguir compila sem fixar a variável p dentro de uma instrução fixed
separada:
class C
{
static S s = new S();
A variável p acessa um elemento em myFixedField . Não é necessário declarar uma variável int* separada. Você
ainda pode precisar de um contexto unsafe . Em versões anteriores do C#, é necessário declarar um segundo
ponteiro fixo:
class C
{
static S s = new S();
Para obter mais informações, confira o artigo sobre como a instrução fixed .
É possível reatribuir variáveis locais ref
Agora, as variáveis locais ref podem ser reatribuídas para se referir a diferentes instâncias depois de serem
inicializadas. O comando a seguir agora compila:
Para saber mais, confira o artigo sobre retornos ref e locais ref e o artigo sobre foreach .
Matrizes stackalloc são compatíveis com inicializadores
Você conseguiu especificar os valores dos elementos em uma matriz ao inicializá-la:
Agora, essa mesma sintaxe pode ser aplicada a matrizes declaradas com stackalloc :
A instrução fixed era compatível com um conjunto limitado de tipos. A partir do C# 7.3, qualquer tipo que
contenha um método GetPinnableReference() que retorna ref T ou ref readonly T pode ser fixed . A adição
desse recurso significa que fixed pode ser usado com System.Span<T> e tipos relacionados.
Para saber mais, confira o artigo Instrução fixed na referência da linguagem.
Restrições genéricas aprimoradas
Agora é possível especificar o tipo System.Enum ou System.Delegate como restrições de classe base para um
parâmetro de tipo.
Você também pode usar a nova restrição unmanaged , para especificar que um parâmetro de tipo deve ser um tipo
não- gerenciadonão anulável.
Para saber mais, confira os artigos sobre restrições genéricas where e Restrições a parâmetros de tipo.
Adicionar essas restrições a tipos existentes é uma alteração incompatível. Tipos genéricos fechados podem não
atender mais a essas novas restrições.
Os tipos de tupla de C# agora são compatíveis com == e != . Para obter mais informações, confira a seção que
aborda igualdade no artigo sobre tuplas.
Anexar atributos aos campos de suporte de propriedades autoimplementadas
Agora há suporte para esta sintaxe:
[field: SomeThingAboutFieldAttribute]
public int SomeProperty { get; set; }
Quando o modificador de argumento in era adicionado, estes dois métodos causariam ambiguidade:
Agora, a sobrecarga por valor (primeira no exemplo anterior) é melhor do que a versão da referência somente
leitura. Para chamar a versão com o argumento de referência somente leitura, inclua o modificador in ao chamar
o método.
NOTE
Isso foi implementado como uma correção de bug. Não é mais ambíguo nem mesmo com a versão da linguagem definida
como "7.2".
public class B
{
public B(int i, out int j)
{
j = i;
}
}
public class D : B
{
public D(int i) : base(i, out var j)
{
Console.WriteLine($"The value of 'j' is {j}");
}
}
O C# 7.2 é outra versão de ponto que adiciona vários recursos úteis. Um tema desta versão é o trabalho com
maior eficiência com tipos de valor, evitando cópias ou alocações desnecessárias.
Os recursos restantes são pequenos e agradáveis.
O C# 7.2 usa o elemento de configuração de seleção de versão de linguagem para selecionar a versão de
linguagem do compilador.
Os novos recursos de linguagem nesta versão são:
Técnicas para escrever código eficiente seguro
Uma combinação de aprimoramentos de sintaxe que permitem trabalhar com tipos de valor usando a
semântica de referência.
Argumentos nomeados que não estejam à direita
Os argumentos nomeados podem ser seguidos por argumentos posicionais.
Sublinhados à esquerda em literais numéricos
Agora os literais numéricos podem ter sublinhados à esquerda, antes dos dígitos impressos.
Modificador de acesso private protected
O modificador de acesso private protected permite o acesso a classes derivadas no mesmo assembly.
Expressões ref condicionais
O resultado de uma expressão condicional ( ?: ) agora já pode ser uma referência.
O restante deste artigo fornece uma visão geral de cada recurso. Para cada recurso, você aprenderá o raciocínio
por trás dele. Você aprenderá a sintaxe. Você pode explorar esses recursos em seu ambiente usando a ferramenta
global dotnet try :
1. Instale a ferramenta global dotnet-try.
2. Clone o repositório dotnet/try-samples.
3. Definir o diretório atual do subdiretório csharp7 para o repositório try-samples.
4. Execute dotnet try .
O C# 7.1 é a primeira versão de ponto da linguagem C#. Ele marca uma cadência de versão maior para a
linguagem. Use os novos recursos mais cedo, de preferência, quando cada novo recurso estiver pronto. O C# 7.1
adiciona a capacidade de configurar o compilador para que ele corresponda a uma versão especificada da
linguagem. Isso permite que você separe a decisão de atualizar ferramentas da decisão de atualizar versões da
linguagem.
O C# 7.1 adiciona a seleção de versão da linguagem elemento de configuração, três novos recursos de linguagem
e um novo comportamento do compilador.
Os novos recursos de linguagem nesta versão são:
async Método Main
O ponto de entrada para um aplicativo pode ter o modificador async .
default Expressões literais
Use expressões literais padrão em expressões de valor padrão quando o tipo de destino pode ser
inferido.
Nomes de elementos de tupla inferidos
Em muitos casos, os nomes dos elementos de tupla podem ser inferidos com base na inicialização da
tupla.
Correspondência de padrões em parâmetros de tipo genérico
Você pode usar expressões de correspondência de padrão em variáveis cujo tipo é um parâmetro de tipo
genérico.
Por fim, o compilador traz duas opções -refout e -refonly , que controlam a geração de assembly de referência.
Para usar as últimas funcionalidades em uma versão de ponto, você precisa configurar a versão da linguagem do
compilador e selecionar a versão.
O restante deste artigo fornece uma visão geral de cada recurso. Para cada recurso, você aprenderá o raciocínio
por trás dele. Você aprenderá a sintaxe. Você pode explorar esses recursos em seu ambiente usando a ferramenta
global dotnet try :
1. Instale a ferramenta global dotnet-try.
2. Clone o repositório dotnet/try-samples.
3. Definir o diretório atual do subdiretório csharp7 para o repositório try-samples.
4. Execute dotnet try .
Async main
Um método async main permite que você use await no método Main . Anteriormente, você precisava escrever:
Se o programa não retorna um código de saída, declare um método Main que retorna um Task:
Para saber mais, confira a seção Literais padrão do artigo do operador padrão.
int count = 5;
string label = "Colors used in the map";
var pair = (count: count, label: label);
Os nomes dos elementos de tupla podem ser inferidos com base nas variáveis usadas para inicializar a tupla no C#
7.1:
int count = 5;
string label = "Colors used in the map";
var pair = (count, label); // element names are "count" and "label"
Variáveis out
A sintaxe existente que dá suporte a parâmetros out foi aperfeiçoada nesta versão. Agora você pode declarar
variáveis out na lista de argumentos de uma chamada de método, em vez de escrever uma instrução de
declaração separada:
if (int.TryParse(input, out int result))
Console.WriteLine(result);
else
Console.WriteLine("Could not parse input");
Você talvez queira especificar o tipo da variável out para maior clareza, conforme mostrado acima. No entanto, a
linguagem dá suporte ao uso de variável local de tipo implícito:
Tuplas
O C# fornece uma sintaxe avançada para classes e structs que são usados para explicar a intenção do design. Mas,
às vezes, essa sintaxe avançada requer trabalho adicional com poucas vantagens. Geralmente, você pode escrever
métodos que precisam de uma estrutura simples que contém mais de um elemento de dados. Para dar suporte a
esses cenários foram adicionadas tuplas ao C#. As tuplas são estruturas de dados leves que contêm vários campos
para representar os membros de dados. Os campos não são validados, e você não pode definir seus próprios
métodos
NOTE
As tuplas estavam disponíveis antes do C# 7.0, mas elas eram ineficientes e não tinham nenhum suporte de linguagem. Isso
significava que os elementos de tupla só podiam ser referenciados como Item1 , Item2 e assim por diante. O C# 7.0
introduz o suporte de linguagem para tuplas, que permite nomes semânticos para os campos de uma tupla, usando tipos de
tupla novos e mais eficientes.
Você pode criar uma tupla atribuindo um valor a cada membro e, opcionalmente, fornecendo nomes semânticos
para cada um dos membros da tupla:
A tupla namedLetters contém campos denominados Alpha e Beta . Esses nomes existem somente em tempo de
compilação e não são preservados, por exemplo, ao inspecionar a tupla usando a reflexão em tempo de execução.
Em uma atribuição de tupla, você também pode especificar os nomes dos campos no lado direito da atribuição:
Pode haver ocasiões em que você deseja descompactar os membros de uma tupla que foram retornados de um
método. Você pode fazer isso declarando variáveis separadas para cada um dos valores na tupla. Essa
descompactação é chamada desconstrução da tupla:
(int max, int min) = Range(numbers);
Console.WriteLine(max);
Console.WriteLine(min);
Você também pode fornecer uma desconstrução semelhante para qualquer tipo no .NET. Escreva um método
Deconstruct como um membro da classe. esse método Deconstruct fornece um conjunto de argumentos out
para cada uma das propriedades que você deseja extrair. Considere essa classe Point que fornece um método
desconstrutor que extrai as coordenadas X e Y :
Descartes
Geralmente, ao desconstruir uma tupla ou chamar um método com parâmetros out , você é forçado a definir uma
variável cujo valor não é importante e você não pretende usar. O C# adiciona suporte para descartes para lidar com
esse cenário. Um descarte é uma variável de somente gravação cujo nome é _ (o caractere de sublinhado); você
pode atribuir todos os valores que você pretende descartar à variável única. Um descarte é como uma variável não
atribuída. Além da instrução de atribuição, o descarte não pode ser usado no código.
Os descartes são compatíveis com os seguintes cenários:
Ao desconstruir tuplas ou tipos definidos pelo usuário.
Ao chamar métodos com parâmetros out.
Em uma operação de correspondência de padrões com as instruções is e switch.
Como um identificador autônomo quando você deseja identificar explicitamente o valor de uma atribuição
como um descarte.
O exemplo a seguir define um método QueryCityDataForYears que retorna uma tupla de 6 que contém dados de
dois anos diferentes para uma cidade. A chamada do método no exemplo é relacionada somente com os dois
valores de população retornados pelo método e, por isso, trata os valores restantes na tupla como descartes ao
desconstruir a tupla.
using System;
using System.Collections.Generic;
private static (string, double, int, int, int, int) QueryCityDataForYears(string name, int year1, int
year2)
{
int population1 = 0, population2 = 0;
double area = 0;
Correspondência de padrões
Correspondência de padrões é um recurso que permite que você implemente a expedição do método em
propriedades diferentes do tipo de um objeto. Você provavelmente já está familiarizado com a expedição de
método com base no tipo de um objeto. Na programação orientada a objeto, os métodos virtuais e de substituição
fornecem a sintaxe da linguagem para implementar a expedição de método com base no tipo de um objeto. As
classes base e derivada fornecem implementações diferentes. As expressões de correspondência de padrões
estendem esse conceito, de modo que você possa implementar com facilidade padrões de expedição semelhantes
para elementos de dados e tipos que não são relacionados por meio de uma hierarquia de herança.
A correspondência de padrões tem suporte a expressões is e switch . Cada uma delas permite inspecionar um
objeto e suas propriedades para determinar se esse objeto satisfaz o padrão procurado. Você usa a palavra-chave
when para especificar regras adicionais para o padrão.
A expressão de padrão is estende o operador is conhecido para consultar um objeto sobre seu tipo e atribuir o
resultado em uma única instrução. O seguinte código verifica se uma variável é um int e, nesse caso, adiciona-a à
soma atual:
Você pode declarar o valor retornado como uma ref e modificar esse valor na matriz, conforme mostrado no
seguinte código:
A linguagem C# tem várias regras que protegem contra o uso indevido de locais e retornos de ref :
É necessário adicionar a palavra-chave ref à assinatura do método e a todas as instruções return em um
método.
Isso torna claro que o método é retornado por referência em todo o método.
Um ref return pode ser atribuído a uma variável de valor ou a uma variável ref .
O chamador controla se o valor retornado é copiado ou não. A omissão do modificador ref ao atribuir
o valor retornado indica que o chamador deseja obter uma cópia do valor, não uma referência ao
armazenamento.
Não é possível atribuir um valor retornado do método padrão a uma variável local de ref .
Isso proíbe que instruções como ref int i = sequence.Count();
Não é possível retornar um ref para uma variável cujo tempo de vida não se estende para além da execução
do método.
Isso significa que não é possível retornar uma referência a uma variável local ou a uma variável com um
escopo semelhante.
O locais e retornos de ref não podem ser usados com métodos assíncronos.
O compilador não consegue saber se a variável referenciada foi definida com o valor final quando o
método assíncrono retorna.
A adição de locais e retornos de ref permite algoritmos que são mais eficientes evitando a cópia de valores ou a
execução múltipla de operações de desreferenciamento.
Adicionar ref ao valor retornado é uma alteração compatível com a origem. O código existente é compilado, mas
o valor retornado ref é copiado quando atribuído. Os chamadores devem atualizar o armazenamento para o valor
retornado para uma variável local ref para armazenar o retorno como uma referência.
Para saber mais, confira o artigo Palavra-chave ref.
Funções locais
Muitos designs para classes incluem métodos que são chamados de apenas um local. Esses métodos privados
adicionais mantêm cada método pequeno e focado. As funções locais permitem que você declare métodos dentro
do contexto de outro método. Funções locais tornam mais fácil para os leitores da classe verem que o método local
é chamado apenas do contexto em que é declarado.
Há dois casos de uso muito comuns para funções locais: métodos iteradores públicos e métodos assíncronos
públicos. Ambos os tipos de métodos de geram código que relata os erros mais tarde do que os programadores
podem esperar. Em métodos iteradores, as exceções são observadas apenas ao chamar o código que enumera a
sequência retornada. Em métodos assíncronos, as exceções são observadas apenas quando a Task retornada é
aguardada. O seguinte exemplo demonstra a validação de parâmetro de separação da implementação do iterador
usando uma função local:
return alphabetSubsetImplementation();
IEnumerable<char> alphabetSubsetImplementation()
{
for (var c = start; c < end; c++)
yield return c;
}
}
A mesma técnica pode ser empregada com métodos async para garantir que as exceções decorrentes da
validação de argumento sejam lançadas antes de o trabalho assíncrono começar:
return longRunningWorkImplementation();
NOTE
Alguns dos designs com suporte pelas funções locais também podem ser feitos usando expressões lambda. Os interessados
podem ler mais sobre as diferenças
// Expression-bodied finalizer
~ExpressionMembersExample() => Console.Error.WriteLine("Finalized!");
NOTE
Este exemplo não precisa de um finalizador, mas ele é mostrado para demonstrar a sintaxe. Você não deve implementar um
finalizador em sua classe a menos que seja necessário para liberar recursos não gerenciados. Você também deve considerar o
uso da classe SafeHandle em vez de gerenciar recursos não gerenciados diretamente.
Esses novos locais para membros aptos para expressão representam um marco importante para a linguagem C#:
Esses recursos foram implementados por membros da comunidade que trabalham no projeto Roslyn de software
livre.
A alteração de um método para um membro de corpo da expressão é uma alteração compatível com binário.
Expressões throw
No C#, throw sempre foi uma instrução. Como throw é uma instrução, não uma expressão, havia constructos do
C# em que não era possível usá-la. Eles incluíam expressões condicionais, expressões de união nulas e algumas
expressões lambda. A adição de membros aptos para expressão inclui mais locais em que as expressões throw
seriam úteis. Para que você possa escrever qualquer um desses construtos, o C# 7.0 apresenta expressões throw.
Essa adição facilita a escrita de um código mais baseado em expressão. Você não precisa de instruções adicionais
para a verificação de erros.
Essa melhoria é mais útil para autores de biblioteca impedirem a alocação de uma Task em um código crítico para
o desempenho.
O 0b no início da constante indica que o número é escrito como um número binário. Números binários podem
ficar longos e, portanto, é mais fácil ver os padrões de bit introduzindo o _ como um separador de dígito,
conforme mostrado acima na constante binária. O separador de dígitos pode aparecer em qualquer lugar na
constante. Para números de base 10, é comum usá-lo como um separador de milhar:
O separador de dígito pode ser usado com os tipos decimal , float e double também:
Juntos, você pode declarar constantes numéricas com muito mais facilidade de leitura.
Novidades no C# 6
23/10/2019 • 17 minutes to read • Edit Online
A versão 6.0 do C# tinha muitos recursos para melhorar a produtividade para desenvolvedores. O efeito geral
desses recursos é que você escreve código mais conciso e também mais legível. A sintaxe contém menos
cerimônia para várias práticas comuns. É mais fácil de ver a intenção do design com menos cerimônia. Aprenda
bem esses recursos para ser mais produtivo e escrever um código mais legível. Você pode se concentrar mais nos
recursos do que nos constructos da linguagem.
O restante deste artigo oferece uma visão geral de cada um desses recursos, com um link para explorá-los.
Explore também os recursos em uma exploração interativa sobre o C# 6 na seção de tutoriais.
As propriedades FirstName e LastName podem ser definidas apenas no corpo do construtor da mesma classe:
Esse recurso habilita o suporte real à linguagem para criar tipos imutáveis e usar a sintaxe de propriedade
automática mais concisa e conveniente.
Se a adição dessa sintaxe não remover um método acessível, essa será uma alteração compatível com binário.
O membro Grades é inicializado no local em que é declarado. Isso facilita realizar a inicialização exatamente uma
vez. A inicialização faz parte da declaração de propriedade, tornando mais fácil igualar a alocação de
armazenamento com a interface pública para objetos Student .
Você também pode usar essa sintaxe para propriedades somente leitura:
A alteração de um membro existente para um membro de corpo da expressão é uma alteração compatível com
binário.
usando estático
O aprimoramento using static permite que você importe os métodos estáticos de uma única classe. Você
especifica a classe que está usando:
O Math não contém nenhum método de instância. Você também pode usar using static para importar os
métodos estáticos de uma classe para uma classe que tem os métodos estáticos e de instância. Um dos exemplos
mais úteis é String:
NOTE
Você precisa usar o nome de classe totalmente qualificado, System.String em uma instrução using estática. Não é possível
usar a palavra-chave string em vez disso.
Quando importados de uma instrução static using , os métodos de extensão apenas estão no escopo quando
chamados usando a sintaxe de invocação de método de extensão. Eles não estão no escopo quando chamados
como um método estático. Você verá isso com frequência em consultas LINQ. Você pode importar o padrão LINQ
importando Enumerable ou Queryable.
Normalmente, os métodos de extensão são chamados usando expressões de invocação de método de extensão.
No caso raro em que você os chama usando a sintaxe de chamada de método estático, adicione o nome de classe
para resolver a ambiguidade.
A diretiva static using também importa todos tipos aninhados. Você pode referenciar qualquer tipo aninhado
sem qualificação.
No exemplo anterior, a variável first é atribuída como null se o objeto person é null . Caso contrário, ele
receberá o valor da propriedade FirstName . Mais importante, o ?. significa que essa linha de código não gera
uma NullReferenceException quando a variável person é null . Em vez disso, ele causa um curto-circuito e
retorna null . Você também pode usar um operador condicional nulo para acesso de matriz ou de indexador.
Substitua [] por ?[] na expressão do índice.
A seguinte expressão retorna um string , independentemente do valor de person . Geralmente esse constructo é
usado com o operador de união nulo para atribuir valores padrão quando uma das propriedades é null . Quando
a expressão causa um curto-circuito, o valor de null retornado é tipado para corresponder à expressão completa.
Você também pode usar o ?. para invocar métodos condicionalmente. O uso mais comum das funções de
membro com o operador condicional nulo é invocar delegados (ou manipuladores de eventos) com segurança que
podem ser null . Você chamará o método Invoke do delegado usando o operador ?. para acessar o membro.
Veja um exemplo no artigo padrões de delegado.
As regras do operador ?. garantem que o lado esquerdo do operador é avaliado apenas uma vez. Isso permite
diversas expressões, incluindo o exemplo a seguir usando manipuladores de eventos:
// preferred in C# 6:
this.SomethingHappened?.Invoke(this, eventArgs);
Garantir que o lado esquerdo seja avaliado apenas uma vez também permite que você use qualquer expressão,
inclusive chamadas de método, no lado esquerdo do ?.
Este exemplo usa propriedades para as expressões substituídas. Você pode usar qualquer expressão. Por exemplo,
você pode calcular a média do aluno como parte da interpolação:
A linha de código anterior formata o valor de Grades.Average() como um número de ponto flutuante com duas
casas decimais.
Muitas vezes, pode ser necessário formatar a cadeia de caracteres produzida usando uma cultura específica.
Aproveite o fato de que o objeto produzido por uma interpolação de cadeia de caracteres pode ser convertido
implicitamente em System.FormattableString. A instância de FormattableString contém a cadeia de caracteres de
formato composto e os resultados da avaliação das expressões antes da conversão delas em cadeias de caracteres.
Use o método FormattableString.ToString(IFormatProvider) para especificar a cultura ao formatar uma cadeia de
caracteres. O exemplo a seguir produz uma cadeia de caracteres usando a cultura de-DE (alemão). (Por padrão, a
cultura alemã usa o caractere ',' para o separador decimal e o caractere '.' como o separador de milhares.)
Para familiarizar-se com a interpolação de cadeia de caracteres, confira o tutorial interativo Interpolação de cadeia
de caracteres em C#, o artigo Interpolação de cadeia de caracteres e o tutorial Interpolação de cadeia de caracteres
em C#.
Filtros de exceção
Os Filtros de Exceção são cláusulas que determinam quando uma determinada cláusula catch deve ser aplicada.
Se a expressão usada para um filtro de exceção é avaliada como true , a cláusula catch realiza seu processamento
normal em uma exceção. Se a expressão for avaliada como false , a cláusula catch será ignorada. Um uso é
examinar informações sobre uma exceção para determinar se uma cláusula catch pode processar a exceção:
A expressão nameof
A expressão nameof é avaliada como o nome de um símbolo. É uma ótima maneira de fazer com que as
ferramentas funcionem sempre que você precisar do nome de uma variável, de uma propriedade ou de um campo
de membro. Um dos usos mais comuns para nameof é fornecer o nome de um símbolo que causou uma exceção:
if (IsNullOrWhiteSpace(lastName))
throw new ArgumentException(message: "Cannot be blank", paramName: nameof(lastName));
Outro uso é com aplicativos baseados em XAML que implementam a interface INotifyPropertyChanged :
public string LastName
{
get { return lastName; }
set
{
if (value != lastName)
{
lastName = value;
PropertyChanged?.Invoke(this,
new PropertyChangedEventArgs(nameof(LastName)));
}
}
}
private string lastName;
Os detalhes de implementação para adicionar o suporte ao await dentro das cláusulas catch e finally garante
que o comportamento seja consistente com o comportamento do código síncrono. Quando o código que é
executado em uma cláusula catch ou finally realiza o lançamento, a execução procura por uma cláusula catch
adequada no próximo bloco. Se houver uma exceção atual, essa exceção será perdida. O mesmo acontece com
expressões aguardadas em cláusulas catch e finally : um catch adequado é pesquisado e a exceção atual, se
houver, será perdida.
NOTE
Esse comportamento é a razão pela qual recomenda-se escrever cláusulas catch e finally com cuidado, a fim de evitar
introduzir novas exceções.
Você pode usá-los com coleções Dictionary<TKey,TValue> e outros tipos nos quais o método Add acessível aceite
mais de um argumento. A nova sintaxe permite a atribuição usando um índice na coleção:
Esse recurso significa que os contêineres associativos podem ser inicializados usando uma sintaxe semelhante à
que está em vigor para contêineres de sequência de várias versões.
Em versões anteriores do C#, haveria uma falha ao chamar esse método usando a sintaxe de grupo de método:
Task.Run(DoThings);
Este artigo fornece um histórico de cada versão principal da linguagem C#. A equipe C# continua a inovar e a
adicionar novos recursos. Os status detalhados do recurso de linguagem, incluindo os recursos considerados para
as versões futuras, podem ser encontrados no repositório dotnet/roslyn no GitHub.
IMPORTANT
A linguagem C# depende de tipos e métodos nos quais a especificação C# é definida como uma biblioteca padrão para
alguns dos recursos. A plataforma .NET fornece esses tipos e métodos em alguns pacotes. Um exemplo é o processamento de
exceção. Cada instrução ou expressão throw é verificada para garantir que o objeto que está sendo gerado é derivado de
Exception. Da mesma forma, cada catch é verificado para garantir que o tipo que está sendo capturado é derivado de
Exception. Cada versão pode adicionar novos requisitos. Para usar os recursos de linguagem mais recentes em ambientes
mais antigos, talvez seja necessário instalar bibliotecas específicas. Essas dependências estão documentadas na página de
cada versão específica. Saiba mais sobre as relações entre linguagem e biblioteca para obter informações sobre essa
dependência.
As ferramentas de compilação do C# consideram a versão mais recente da linguagem principal como a versão
padrão da linguagem. Pode haver versões de ponto entre as versões principais, detalhadas em outros artigos nesta
seção. Para usar as últimas funcionalidades em uma versão de ponto, você precisa configurar a versão da
linguagem do compilador e selecionar a versão. Já houve três versões de ponto desde o C# 7.0:
C# 7.3:
O C# 7.3 está disponível a partir do Visual Studio 2017 versão 15.7 e do SDK do .NET Core 2.1.
C# 7.2:
C#7,2 está disponível a partir do Visual Studio 2017 versão 15,5 e do SDK do .NET Core 2,0.
C# 7.1:
O C# 7.1 está disponível a partir do Visual Studio 2017 versão 15.3 e do SDK do .NET Core 2.0.
C# versão 1.0
Quando voltar e olhar, C# a versão 1,0, lançada com o Visual Studio .NET 2002, ficou muito parecida com o Java.
Como parte de suas metas de design declaradas para ECMA, ela buscava ser uma "linguagem simples, moderna,
de uso geral e orientada a objeto". No momento, parece que o Java alcançou essas metas de design iniciais.
Mas agora, se examinar novamente a C# 1.0, você poderá se sentir um pouco confuso. Carecia das funcionalidades
assíncronas internas e algumas das funcionalidades relacionadas a genéricos que você nem valoriza. Na verdade,
ela não tinha nada relacionado a genéricos. E a LINQ? Ainda não estava disponível. Essas adições levariam alguns
anos para sair.
A versão 1.0 do C# parecia ter poucos recursos, em comparação com os dias de hoje. Você teria que escrever
código um tanto detalhado. Mas, ainda assim, você poderia iniciar em algum lugar. A versão 1.0 do C# era uma
alternativa viável ao Java na plataforma Windows.
Os principais recursos do C# 1.0 incluíam:
Classes
Structs
Interfaces
Eventos
Propriedades
Delegados
Expressões
Instruções
Atributos
C# versão 1.2
C#a versão 1,2 foi fornecida com o Visual Studio .NET 2003. Ele continha algumas pequenas melhorias na
linguagem. Muito notável é que, a partir desta versão, o código gerado em um loop foreach chamou Dispose, em
um IEnumerator, quando o IEnumerator implementou IDisposable.
C# versão 2.0
Neste momento, as coisas começam a ficar interessantes. Vamos dar uma olhada em alguns recursos principais do
C# 2.0, lançado em 2005, junto com o Visual Studio 2005:
Genéricos
Tipos parciais
Métodos anônimos
Tipos de valor anuláveis
Iteradores
Covariância e contravariância
Outros recursos do C# 2.0 adicionaram funcionalidades a recursos existentes:
Acessibilidade separada getter/setter
Conversões de grupo de método (delegados)
Classes estáticas
Inferência de delegado
Embora C# possa ter começado como uma linguagem OO (orientada a objeto) genérica, a versão 2.0 do C# tratou
de mudar isso rapidamente. Depois de se acostumarem com a ideia da linguagem OO, os desenvolvedores
começaram a sofrer com vários pontos problemáticos graves. E os procuraram de maneira significativa.
Com genéricos, tipos e métodos podem operar em um tipo arbitrário enquanto ainda mantêm a segurança de
tipos. Por exemplo, ter um List<T> permite que você tenha List<string> ou List<int> e execute operações
fortemente tipadas nessas cadeias de caracteres ou inteiros enquanto itera neles. Usar genéricos é melhor que criar
ListInt que deriva de ArrayList ou converter de Object para cada operação.
A versão 2.0 do C# trouxe iteradores. Em resumo, o iteradores permitem que você examine todos os itens em um
List (ou outros tipos Enumeráveis) com um loop foreach . Ter iteradores como uma parte importante da
linguagem aprimorou drasticamente a legibilidade da linguagem e a capacidade das pessoas de raciocinar a
respeito do código.
E ainda assim, o C# continuava na tentativa de alcançar o mesmo nível do Java. O Java já tinha liberado versões
que incluíam genéricos e iteradores. Mas isso seria alterado logo, pois as linguagens continuaram a evoluir
separadamente.
C# versão 3.0
O C# versão 3.0 chegou no final de 2007, juntamente com o Visual Studio 2008, porém o pacote completo de
recursos de linguagem veio, na verdade, com o C# versão 3.5. Esta versão foi o marco de uma alteração
significativa no crescimento do C#. Ela estabeleceu o C# como uma linguagem de programação realmente
formidável. Vamos dar uma olhada em alguns recursos importantes nesta versão:
Propriedades autoimplementadas
Tipos anônimos
Expressões de consulta
Expressões lambda
Árvores de expressão
Métodos de extensão
Variáveis locais implicitamente tipadas
Métodos parciais
Inicializadores de objeto e de coleção
Numa retrospectiva, muitos desses recursos parecerem inevitáveis e inseparáveis. Todos eles se encaixam
estrategicamente. Costuma-se pensar que o recurso irresistível dessa versão do C# foi a expressão de consulta,
também conhecida como LINQ (consulta integrada à linguagem).
Uma exibição mais detalhada analisa árvores de expressão, expressões lambda e tipos anônimos como a base na
qual o LINQ é construído. Mas, de uma forma ou de outra, o C# 3.0 apresentou um conceito revolucionário. O C#
3.0 começou a definir as bases para transformar o C# em uma linguagem híbrida orientada a objeto e funcional.
Especificamente, agora você pode escrever consultas declarativas no estilo SQL para executar operações em
coleções, entre outras coisas. Em vez de escrever um loop for para calcular a média de uma lista de inteiros,
agora você pode fazer isso simplesmente como list.Average() . A combinação de expressões de consulta e
métodos de extensão fez parecer que essa lista de inteiros se tornasse muito mais inteligente.
Levou algum tempo para que as pessoas entendessem e integrassem o conceito, mas isso aconteceu
gradualmente. E agora, anos mais tarde, o código é muito mais conciso, simples e funcional.
C# versão 4.0
C#a versão 4,0, lançada com o Visual Studio 2010, teria tido um tempo difícil de viver até o status inovador da
versão 3,0. Com a versão 3.0, o C# tirou verdadeiramente a linguagem da sombra do Java e a colocou em
proeminência. A linguagem foi rapidamente se tornando elegante.
A próxima versão introduziu alguns novos recursos interessantes:
Associação dinâmica
Argumentos opcionais/nomeados
Genérico covariante e contravariante
Tipos de interoperabilidade inseridos
Os tipos de interoperabilidade inseridos atenuaram um problema de implantação. A contravariância e a
covariância genérica oferecem maior capacidade para usar genéricos, mas eles são um tanto acadêmicos e
provavelmente mais apreciados por autores de estruturas e bibliotecas. Os parâmetros nomeados e opcionais
permitem eliminar várias sobrecargas de método e oferecem conveniência. Mas nenhum desses recursos é
exatamente uma alteração de paradigma.
O recurso principal foi a introdução da palavra-chave dynamic . A palavra-chave dynamic introduziu na versão 4.0
do C# a capacidade de substituir o compilador na tipagem em tempo de compilação. Com o uso da palavra-chave
dinâmica, você pode criar constructos semelhantes a linguagens dinamicamente tipadas, como JavaScript. Você
pode criar um dynamic x = "a string" e, em seguida, adicionar seis a ela, deixando que o runtime decida o que
acontece em seguida.
Associação dinâmica tem potencial de erros, mas também grande eficiência na linguagem.
C# versão 5.0
C#a versão 5,0, lançada com o Visual Studio 2012, foi uma versão focada da linguagem. Quase todo o esforço para
essa versão foi dedicado a outro conceito inovador de linguagem: os modelos async e await para programação
assíncrona. Aqui está a lista dos recursos principais:
Membros assíncronos
Atributos de informações do chamador
Consulte também
Code Project: Caller Info Attributes in C# 5.0 (Code Project: Atributos de informações do chamador em C# 5.0)
O atributo de informações do chamador permite facilmente recuperar informações sobre o contexto no qual você
está executando sem recorrer a uma infinidade de código de reflexão clichê. Ele tem muitos usos em diagnóstico e
tarefas de registro em log.
Mas async e await são as verdadeiras estrelas dessa versão. Quando esses recursos foram lançados em 2012, o
C# virou o jogo novamente, implantando a assincronia na linguagem como uma participante da maior
importância. Se você já teve que lidar com operações de longa execução e a implementação de redes de retorno de
chamada, você provavelmente adorou esse recurso de linguagem.
C# versão 6.0
Nas versões 3.0 e 5.0, o C# recebeu alguns novos recursos importantes em uma linguagem orientada a objeto.
Com a versão 6,0, lançada com o Visual Studio 2015, ela desapareceria com um recurso de Killer dominante e, em
vez disso, C# lançaria muitos recursos menores que tornaram a programação mais produtiva. Aqui estão alguns
deles:
Importações estáticas
Filtros de exceção
Inicializadores de propriedade automática
Membros aptos para expressão
Propagador nulo
Interpolação de cadeia de caracteres
Operador nameof
Inicializadores de índice
Outros novos recursos incluem:
Await em blocos catch/finally
Valores padrão para propriedades somente getter
Cada um desses recursos é interessante em seus próprios méritos. Mas, se você os observar em conjunto, verá um
padrão interessante. Nesta versão, o C# eliminou o clichê de linguagem para tornar o código mais conciso e
legível. Portanto, para os fãs de código simples e conciso, essa versão da linguagem foi um grande benefício.
Fizeram ainda outra coisa com esta versão, embora não seja um recurso de linguagem tradicional em si. Lançaram
Roslyn, o compilador como um serviço. Agora o compilador de C# é escrito em C#, e você pode usar o compilador
como parte de seus esforços de programação.
C# versão 7.0
A versão principal mais recente é C# a versão 7,0, lançada com o Visual Studio 2017. Esta versão tem algumas
coisas interessantes e evolutivas na mesma direção que o C# 6.0, mas sem o compilador como um serviço. Aqui
estão alguns dos novos recursos:
Variáveis Out
Tuplas e desconstrução
Correspondência de padrões
Funções locais
Membros aptos para expressão expandidos
Locais e retornos de ref
Outros recursos incluíam:
Descarta
Literais binários e os separadores de dígito
Expressões throw
Todas essas funcionalidades oferecem novos recursos interessantes para desenvolvedores e a oportunidade de
escrever um código mais limpo do que nunca. Um ponto alto é a condensação da declaração de variáveis a serem
usadas com a palavra-chave out e a permissão de vários valores retornados por meio de tupla.
Mas o C# está sendo colocado para um uso cada vez mais amplo. Agora o .NET Core tem qualquer sistema
operacional como destino e tem a visão firme na nuvem e na portabilidade. Essas novas funcionalidades
certamente ocupam a mente e o tempo dos designers da linguagem, além de levarem a novos recursos.
Artigo originalmente publicado no blog NDepend , cortesia de Erik Dietrich e Patrick Smacchia.
Relações entre os recursos de linguagem e os tipos
de bibliotecas
22/06/2018 • 3 minutes to read • Edit Online
A definição de linguagem C# exige que uma biblioteca padrão tenha determinados tipos e determinados
membros acessíveis nesses tipos. O compilador gera o código que usa esses tipos e membros necessários para
muitos recursos de linguagem diferentes. Quando necessário, há pacotes NuGet que contêm os tipos necessários
para as versões mais recentes da linguagem ao escrever um código para ambientes em que esses tipos ou
membros ainda não foram implantados.
Essa dependência da funcionalidade da biblioteca padrão faz parte da linguagem C# desde sua primeira versão.
Nessa versão, os exemplos incluíam:
Exception – usado para todas as exceções geradas pelo compilador.
String – tipo string C# é um sinônimo de String.
Int32 – sinônimo de int .
A primeira versão era simples: o compilador e a biblioteca padrão eram fornecidos juntos e havia somente uma
versão de cada um.
As versões posteriores do C# ocasionalmente adicionaram novos tipos ou membros às dependências. Os
exemplos incluem: INotifyCompletion, CallerFilePathAttribute e CallerMemberNameAttribute. O C# 7.0 continua
isso adicionando uma dependência de ValueTuple para implementar o recurso de linguagem de tuplas.
A equipe de design de linguagem trabalha para minimizar a área de superfície dos tipos e membros necessários
em uma biblioteca padrão em conformidade. Essa meta é equilibrada em relação a um design limpo no qual
novos recursos de biblioteca são incorporados diretamente na linguagem. Haverá novos recursos em versões
futuras do C# que exigem novos tipos e membros em uma biblioteca padrão. É importante entender como
gerenciar essas dependências em seu trabalho.
Gerenciando as dependências
As ferramentas do compilador C# agora são desacopladas do ciclo de liberação das bibliotecas .NET em
plataformas com suporte. Na verdade, bibliotecas .NET diferentes têm ciclos de liberação diferentes: o .NET
Framework no Windows é liberado como um Windows Update, o .NET Core é fornecido em um agendamento
separado e as versões do Xamarin das atualizações de biblioteca são fornecidas com as ferramentas do Xamarin
para cada plataforma de destino.
Na maior parte do tempo, você não perceberá essas alterações. No entanto, quando estiver trabalhando com uma
versão mais recente da linguagem que exige recursos que ainda não estão nas bibliotecas .NET nessa plataforma,
você referenciará os pacotes NuGet para fornecer esses novos tipos. Conforme as plataformas para as quais seu
aplicativo dá suporte forem atualizadas com as novas instalações de estrutura, você poderá remover a referência
extra.
Essa separação significa que você pode usar os novos recursos de linguagem até mesmo quando direcionar
computadores que podem não ter a estrutura correspondente.
Considerações sobre versão e atualização para os
desenvolvedores de C#
23/10/2019 • 4 minutes to read • Edit Online
A compatibilidade é uma meta muito importante quando novos recursos são adicionados à linguagem C#. Em
quase todos os casos, o código existente pode ser recompilado com uma nova versão do compilador sem nenhum
problema.
Mais cuidado pode ser necessário quando você adota novos recursos de linguagem de programação em uma
biblioteca. Você poderá estar criando uma nova biblioteca com recursos encontrados na versão mais recente, e
precisará garantir que os aplicativos criados com versões anteriores do compilador possam usá-la. Ou você pode
estar atualizando uma biblioteca existente e muitos dos seus usuários talvez não tenham atualizado as versões
ainda. Ao tomar decisões sobre a adoção de novos recursos, você precisará considerar duas variações de
compatibilidade: compatível com a origem e compatível com binário.
Alterações incompatíveis
Se uma alteração não for nem compatível com a origem nem compatível com binário, alterações de código-
fonte, juntamente com a recompilação, serão necessárias nas bibliotecas e aplicativos dependentes.
Avaliar a biblioteca
Esses conceitos de compatibilidade afetam as declarações públicas e protegidas para sua biblioteca, mas não a
respectiva implementação interna. A adoção de quaisquer novos recursos internamente é sempre compatível
com binário.
Alterações compatíveis com binário oferecem uma nova sintaxe que gera o mesmo código compilado para
declarações públicas que a sintaxe antiga. Por exemplo, a alteração de um método para um membro de corpo da
expressão é uma alteração compatível com binário:
Código original:
Alterações compatíveis com a origem introduzem sintaxe que altera o código compilado para um membro
público, mas de uma forma compatível com sites de chamada já existentes. Por exemplo, alterar a assinatura de um
método de um parâmetro por valor para um in por referência é compatível com a origem, mas não é compatível
com binário:
Código original:
Novo código:
Os artigos Novidades apontam que a introdução de um recurso que afeta as declarações públicas é compatível
com a origem ou compatível com binário.
Tipos (Guia de Programação em C#)
29/11/2019 • 22 minutes to read • Edit Online
int a = 5;
int b = a + 2; //OK
// Error. Operator '+' cannot be applied to operands of type 'int' and 'bool'.
int c = a + test;
NOTE
Desenvolvedores de C e C++, observem que, em C#, bool não é conversível em int.
O compilador insere as informações de tipo no arquivo executável como metadados. O CLR (Common Language
Runtime) usa metadados em tempo de execução para garantir mais segurança de tipos quando aloca e recupera a
memória.
Especificando tipos em declarações de variável
Quando declara uma variável ou constante em um programa, você deve especificar seu tipo ou usar a palavra-
chave var para permitir que o compilador infira o tipo. O exemplo a seguir mostra algumas declarações de
variáveis que usam tipos numéricos internos e tipos complexos definidos pelo usuário:
// Declaration only:
float temperature;
string name;
MyClass myClass;
Os tipos de parâmetros de método e valores de retorno são especificados na assinatura do método. A assinatura a
seguir mostra um método que requer um int como um argumento de entrada e retorna uma cadeia de caracteres:
Depois que uma variável é declarada, ela não pode ser declarada novamente com um novo tipo e não pode ter um
valor atribuído que não seja compatível com seu tipo declarado. Por exemplo, você não pode declarar um int e, em
seguida, atribuir a ele um valor booliano de true . No entanto, os valores podem ser convertidos em outros tipos,
por exemplo, quando são passados como argumentos de método ou atribuídos a novas variáveis. Uma conversão
de tipo que não causa a perda de dados é executada automaticamente pelo compilador. Uma conversão que pode
causar perda de dados requer um cast no código-fonte.
Para obter mais informações, consulte Conversões e conversões de Tipo.
Tipos internos
O C# fornece um conjunto padrão de tipos numéricos internos para representar números inteiros, valores de ponto
flutuante, expressões boolianas, caracteres de texto, valores decimais e outros tipos de dados. Também há tipos
string e object internos. Eles estão disponíveis para uso em qualquer programa em C#. Para obter mais
informações sobre os tipos internos, consulte Tabelas de referência de tipos internos.
Tipos personalizados
Você usa os constructos struct, classe, interface e enum para criar seus próprios tipos personalizados. A biblioteca
de classes do .NET em si é uma coleção de tipos personalizados fornecida pela Microsoft, que você pode usar em
seus próprios aplicativos. Por padrão, os tipos usados com mais frequência na biblioteca de classes estão
disponíveis em qualquer programa em C#. Outros ficam disponíveis somente quando você adiciona explicitamente
uma referência de projeto ao assembly no qual eles estão definidos. Após o compilador ter uma referência ao
assembly, você pode declarar variáveis (e constantes) dos tipos declarados no assembly no código-fonte. Para saber
mais, confira Biblioteca de classes do .NET.
NOTE
Você pode ver que os tipos mais usados normalmente são todos organizados no namespace System. No entanto, o
namespace no qual um tipo está contido não tem relação com a possibilidade de ele ser um tipo de valor ou um tipo de
referência.
Tipos de valor
Os tipos de valor derivam de System.ValueType, que deriva de System.Object. Os tipos que derivam de
System.ValueType apresentam um comportamento especial no CLR. As variáveis de tipo de valor contêm
diretamente seus valores, o que significa que a memória é alocada embutida em qualquer contexto em que a
variável é declarada. Não há nenhuma alocação de heap separada ou sobrecarga de coleta de lixo para variáveis do
tipo de valor.
Há duas categorias de tipos de valor: struct e enum.
Os tipos numéricos internos são structs e têm propriedades e métodos que você pode acessar:
// Static method on type byte.
byte b = byte.MaxValue;
Mas você declara e atribui valores a eles como se fossem tipos de não agregação simples:
Tipos de valor são lacrados, o que significa, por exemplo, que você não pode derivar um tipo de System.Int32 e não
pode definir um struct para herdar de qualquer struct ou classe definida pelo usuário, porque um struct apenas
pode herdar de System.ValueType. No entanto, um struct pode implementar uma ou mais interfaces. É possível
converter um tipo de struct em qualquer tipo de interface que ele implementa. Isso faz com que uma operação de
conversão boxing envolva o struct dentro de um objeto de tipo de referência no heap gerenciado. As operações de
conversão boxing ocorrem quando você passa um tipo de valor para um método que usa um System.Object ou
qualquer tipo de interface como parâmetro de entrada. Para obter mais informações, consulte Boxing e unboxing.
Você usa a palavra-chave struct para criar seus próprios tipos de valor personalizados. Normalmente, um struct é
usado como um contêiner para um pequeno conjunto de variáveis relacionadas, conforme mostrado no exemplo a
seguir:
Para obter mais informações sobre structs, consulte Structs. Para saber mais sobre os tipos de valor do .NET,
confira Tipos de valor.
A outra categoria de tipos de valor é enum. Uma enum define um conjunto de constantes integrais nomeadas. Por
exemplo, a enumeração System.IO.FileMode na biblioteca de classes do .NET contém um conjunto de números
inteiros constantes nomeados que especificam como um arquivo deve ser aberto. Ela é definida conforme
mostrado no exemplo abaixo:
A constante System.IO.FileMode.Create tem um valor de 2. No entanto, o nome é muito mais significativo para a
leitura do código-fonte por humanos e, por esse motivo, é melhor usar enumerações em vez de números literais
constantes. Para obter mais informações, consulte System.IO.FileMode.
Todas as enumerações herdam de System.Enum, que herda de System.ValueType. Todas as regras que se aplicam a
structs também se aplicam a enums. Para obter mais informações sobre enums, consulte Tipos de enumeração.
Tipos de referência
Um tipo que é definido como uma classe, delegado, matriz ou interface é um tipo de referência. No tempo de
execução, quando você declara uma variável de um tipo de referência, a variável contém o valor null até você criar
explicitamente um objeto usando o operador new ou atribuir a ele um objeto que foi criado em outro lugar com o
operador new , conforme mostrado no exemplo a seguir:
Uma interface deve ser inicializada com um objeto da classe que a implementa. Se MyClass implementa
IMyInterface , você cria uma instância de IMyInterface , conforme mostrado no exemplo a seguir:
Quando o objeto é criado, a memória é alocada no heap gerenciado e a variável contém apenas uma referência
para o local do objeto. Os tipos no heap gerenciado requerem sobrecarga quando são alocados e quando são
recuperados pela funcionalidade de gerenciamento automático de memória do CLR, que é conhecida como coleta
de lixo. No entanto, a coleta de lixo também é altamente otimizada e na maioria dos cenários não cria um problema
de desempenho. Para obter mais informações sobre a coleta de lixo, consulte Gerenciamento automático de
memória.
Todas as matrizes são tipos de referência, mesmo se seus elementos forem tipos de valor. As matrizes derivam
implicitamente da classe System.Array, mas você declara e usa as matrizes com a sintaxe simplificada fornecida
pelo C#, conforme mostrado no exemplo a seguir:
Os tipos de referência dão suporte completo à herança. Ao criar uma classe, você pode herdar de outra interface ou
classe que não está definida como lacrada, e outras classes podem herdar de sua classe e substituir os métodos
virtuais. Para obter mais informações sobre como criar suas próprias classes, consulte Classes e structs. Para obter
mais informações sobre herança e métodos virtuais, consulte Herança.
Tipos genéricos
Um tipo pode ser declarado com um ou mais parâmetros de tipo que servem como um espaço reservado para o
tipo real (o tipo concreto) que o código do cliente fornecerá ao criar uma instância do tipo. Esses tipos são
chamados de tipos genéricos. Por exemplo, o tipo .NET System.Collections.Generic.List<T> tem um parâmetro de
tipo que, por convenção, recebe o nome t. Ao criar uma instância do tipo, você especifica o tipo dos objetos que a
lista conterá, por exemplo, Cadeia de caracteres:
O uso do parâmetro de tipo possibilita a reutilização da mesma classe para conter qualquer tipo de elemento sem
precisar converter cada elemento em objeto. As classes de coleção genéricas são chamadas de coleções fortemente
tipadas porque o compilador sabe o tipo específico dos elementos da coleção e pode gerar um erro em tempo de
compilação se, por exemplo, você tentar adicionar um inteiro ao objeto stringList no exemplo anterior. Para obter
mais informações, consulte Genéricos.
Seções relacionadas
Para mais informações, consulte os seguintes tópicos:
Transmissões e conversões de tipo
Conversão boxing e unboxing
Usando o tipo dynamic
Tipos de valor
Tipos de referência
Classes e Structs
Tipos Anônimos
Genéricos
Especificação da linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a fonte
definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
Conversão de tipos de dados XML
Tipos integrais
Tipos de referência anuláveis
08/11/2019 • 14 minutes to read • Edit Online
O C# 8.0 apresenta tipos de referência que permitem valor nulo e tipos de referência que não permitem
valor nulo que permitem que você crie importantes instruções sobre as propriedades para variáveis de tipo de
referência:
Uma referência não deve ser nula. Quando as variáveis não devem ser nulas, o compilador impõe regras
que garantem que seja seguro desreferenciar essas variáveis sem verificar primeiro que ela não está nula:
A variável deve ser inicializada para um valor não nulo.
A variável nunca pode receber o valor null .
Uma referência pode ser nula. Quando variáveis puderem ser nulas, o compilador imporá diferentes
regras para garantir que você verificou corretamente se há uma referência nula:
a variável só poderá ser desreferenciada quando o compilador puder garantir que o valor não é nulo.
Essas variáveis podem ser inicializadas com o valor null padrão e receber o valor null em outro
código.
Esse novo recurso oferece benefícios significativos com relação à manipulação de variáveis de referência em
versões anteriores do C#, em que a intenção do design não poderia ser determinada na declaração da variável. O
compilador não forneceu segurança com relação a exceções de referência nula para tipos de referência:
Uma referência pode ser nula. Nenhum aviso é emitido quando um tipo de referência é inicializado para
nulo ou atribuído posteriormente a nulo.
Uma referência é considerada não nula. O compilador não emite nenhum aviso quando os tipos de
referência são desreferenciados. (Com referências que permitem valor nulo, o compilador emite avisos
sempre que você desreferencia uma variável que pode ser nula).
Com a adição de tipos de referência que permitem valor nulo, é possível declarar sua intenção mais claramente.
O valor null é a maneira correta de representar que uma variável não se refere a um valor. Não use esse
recurso para remover todos os valores null do seu código. Em vez disso, você deve declarar sua intenção par
ao compilador e para outros desenvolvedores que leem seu código. Ao declarar sua intenção, o compilador
informa quando você escreve um código inconsistente com essa intenção.
Um tipo de referência que permite valor nulo é indicado usando a mesma sintaxe que tipos de valor que
permitem valor nulo: um ? é acrescentado ao tipo da variável. Por exemplo, a seguinte declaração de variável
representa uma variável de cadeia de caracteres que permite valor nulo, name :
string? name;
Qualquer variável em que o ? não é acrescentado ao nome do tipo é um tipo de referência que não permite
valor nulo. Isso inclui todas as variáveis de tipo de referência no código existente quando você habilitou esse
recurso.
O compilador usa uma análise estática para determinar se uma referência que permite valor nulo é conhecida
como não nula. O compilador informa se você desreferencia uma referência quer permite valor nulo quando ela
pode ser nula. Você pode substituir esse comportamento usando o operador NULL -tolerante ! seguindo um
nome de variável. Por exemplo, se você sabe que a variável name não é nula, mas o compilador emite um aviso,
é possível escrever o seguinte código para substituir a análise do compilador:
name!.Length;
<Nullable>enable</Nullable>
Também é possível usar diretivas para definir esses mesmos contextos em qualquer lugar no seu projeto:
#nullable enable : define o contexto de anotação anulável e o contexto de aviso anulável como habilitado.
#nullable disable : define o contexto de anotação anulável e o contexto de aviso anulável como
desabilitado.
#nullable restore : restaura o contexto de anotação anulável e o contexto de aviso anulável para as
configurações do projeto.
#nullable disable warnings : defina o contexto de aviso anulável como desabilitado.
#nullable enable warnings : defina o contexto de aviso anulável como habilitado.
#nullable restore warnings : restaura o contexto de aviso anulável para as configurações do projeto.
#nullable disable annotations : defina o contexto de anotação anulável como desabilitado.
#nullable enable annotations : defina o contexto de anotação anulável como habilitado.
#nullable restore annotations : restaura o contexto de aviso de anotação para as configurações do projeto.
Por padrão, os contextos de anotação e de aviso anuláveis são desabilitados. Isso significa que o código
existente é compilado sem alterações e sem gerar nenhum aviso novo.
Consulte também
Especificação de tipos de referência Nullable de rascunho
Introdução ao tutorial de referências que permitem valor nulo
Migrar uma base de código para referências que permitem valor nulo
Atualizar bibliotecas para usar tipos de referência
anuláveis e comunicar regras anuláveis a chamadores
27/11/2019 • 35 minutes to read • Edit Online
A adição de tipos de referência anuláveis significa que você pode declarar se um valor null ou não é permitido ou
esperado para cada variável. Além disso, você pode aplicar um número de atributos: AllowNull , DisallowNull ,
MaybeNull , NotNull , NotNullWhen , MaybeNullWhen e NotNullWhenNotNull para descrever completamente os Estados
nulos dos valores de argumento e de retorno. Isso fornece uma ótima experiência à medida que você escreve o
código. Você receberá avisos se uma variável não anulável puder ser definida como null . Você receberá avisos se
uma variável anulável não for marcada como nula antes de você desreferenciá-la. Atualizar suas bibliotecas pode
levar tempo, mas os benefícios valem a pena. Quanto mais informações você fornecer ao compilador sobre
quando um valor de null é permitido ou proibido, os melhores avisos que os usuários da sua API receberão.
Vamos começar com um exemplo familiar. Imagine que sua biblioteca tenha a seguinte API para recuperar uma
cadeia de caracteres de recurso:
O exemplo anterior segue o conhecido padrão de Try* no .NET. Há dois argumentos de referência para essa API:
o key e o parâmetro message . Essa API tem as seguintes regras relacionadas à nulidade desses argumentos:
Os chamadores não devem passar null como argumento para key .
Os chamadores podem passar uma variável cujo valor é null como o argumento para message .
Se o método TryGetMessage retornar true , o valor de message não será nulo. Se o valor de retorno for
false, o valor de message (e seu estado nulo) será nulo.
A regra para key pode ser totalmente expressa pelo tipo de variável: key deve ser um tipo de referência não
anulável. O parâmetro message é mais complexo. Ele permite null como o argumento, mas garante que, em caso
de sucesso, que out argumento não seja nulo. Para esses cenários, você precisa de um vocabulário mais rico para
descrever as expectativas.
Atualizar sua biblioteca para referências anuláveis requer mais do que a inundação de ? em algumas das
variáveis e nomes de tipo. O exemplo anterior mostra que você precisa examinar suas APIs e considerar suas
expectativas para cada argumento de entrada. Considere as garantias para o valor de retorno e qualquer out ou
ref argumentos no retorno do método. Em seguida, comunique essas regras ao compilador, e o compilador
fornecerá avisos quando os chamadores não obedecem a essas regras.
Esse trabalho leva tempo. Vamos começar com as estratégias para tornar sua biblioteca ou reconhecimento
anulável de aplicativo, ao mesmo tempo em que equilibra outros requisitos e resultados finais. Você verá como
balancear o desenvolvimento contínuo habilitando tipos de referência anuláveis. Você aprenderá os desafios para
definições de tipo genérico. Você aprenderá a aplicar atributos para descrever as condições anteriores e
posteriores em APIs individuais.
Essa segunda estratégia tem menos trabalho de antecedência. A desvantagem é que a primeira tarefa quando você
cria um novo arquivo é adicionar o pragma e torná-lo indesejado de forma anulável. Se algum desenvolvedor da
sua equipe esquecer, esse novo código estará no registro posterior do trabalho para tornar todos os incompatíveis
com o código anulável.
Qual dessas estratégias você escolhe depende de quanto o desenvolvimento ativo está ocorrendo em seu projeto.
Quanto mais maduro e estável for seu projeto, melhor a segunda estratégia. Quanto mais recursos estiverem
sendo desenvolvidos, melhor será a primeira estratégia.
O valor de retorno indica êxito ou falha e transporta o valor se o valor foi encontrado. Em muitos casos, a alteração
de assinaturas de API pode melhorar a forma como elas comunicam valores nulos.
No entanto, para bibliotecas públicas ou bibliotecas com grandes bases de usuários, você pode preferir não
introduzir nenhuma alteração de assinatura de API. Para esses casos e outros padrões comuns, você pode aplicar
atributos para definir de forma mais clara quando um argumento ou valor de retorno pode ser null . Se você
considerar ou não a alteração da superfície de sua API, provavelmente descobrirá que as anotações de tipo
sozinhas não são suficientes para descrever null valores para argumentos ou valores de retorno. Nessas
instâncias, você pode aplicar atributos para descrever mais claramente uma API.
Quando você compila o código anterior em um contexto alheios anulável, tudo está bem. Depois de habilitar os
tipos de referência anuláveis, a propriedade ScreenName se torna uma referência não anulável. Isso está correto
para o acessador de get : ele nunca retorna null . Os chamadores não precisam verificar a propriedade retornada
para null . Mas, agora, definir a propriedade como null gera um aviso. Para continuar a dar suporte a esse tipo
de código, adicione o atributo System.Diagnostics.CodeAnalysis.AllowNullAttribute à propriedade, conforme
mostrado no código a seguir:
[AllowNull]
public string ScreenName
{
get => screenName;
set => screenName = value ?? GenerateRandomScreenName();
}
private string screenName = GenerateRandomScreenName();
Talvez seja necessário adicionar uma diretiva de using para System.Diagnostics.CodeAnalysis usar esse e outros
atributos discutidos neste artigo. O atributo é aplicado à propriedade, não ao acessador set . O atributo
AllowNull especifica pré -condições e sóse aplica a entradas. O acessador de get tem um valor de retorno, mas
nenhum argumento de entrada. Portanto, o atributo AllowNull só se aplica ao acessador set .
O exemplo anterior demonstra o que procurar ao adicionar o atributo AllowNull em um argumento:
1. O contrato geral para essa variável é que não deve ser null , portanto, você deseja um tipo de referência não
anulável.
2. Há cenários para a variável de entrada ser null , embora não sejam o uso mais comum.
Geralmente, você precisará desse atributo para propriedades, ou in , out e argumentos de ref . O atributo
AllowNull é a melhor opção quando uma variável normalmente é não nula, mas você precisa permitir null como
uma pré-condição.
Compare com cenários para usar DisallowNull : você usa esse atributo para especificar que uma variável de
entrada de um tipo anulável não deve ser null . Considere uma propriedade em que null é o valor padrão, mas
os clientes só podem defini-lo como um valor não nulo. Considere o código a seguir:
O código anterior é a melhor maneira de expressar o design que o ReviewComment poderia ser null , mas não
pode ser definido como null . Depois que esse código é ciente de forma anulável, você pode expressar esse
conceito mais claramente para os chamadores usando o System.Diagnostics.CodeAnalysis.DisallowNullAttribute:
[DisallowNull]
public string? ReviewComment
{
get => _comment;
set => _comment = value ?? throw new ArgumentNullException(nameof(value), "Cannot set to null");
}
string? _comment;
Em um contexto anulável, o acessador de get ReviewComment pode retornar o valor padrão de null . O
compilador avisa que ele deve ser verificado antes do acesso. Além disso, ele avisa aos chamadores que, embora
possa ser null , os chamadores não devem defini-los explicitamente como null . O atributo DisallowNull
também especifica uma pré-condição, ele não afeta o acessador get . Você deve optar por usar o atributo
DisallowNull ao observar essas características sobre:
1. A variável pode ser null em cenários principais, geralmente quando a primeira instância é instanciada.
2. A variável não deve ser definida explicitamente como null .
Essas situações são comuns no código que era originalmente nulo alheios. Pode ser que as propriedades do objeto
sejam definidas em duas operações de inicialização distintas. Pode ser que algumas propriedades sejam definidas
somente após a conclusão de algum trabalho assíncrono.
Os atributos AllowNull e DisallowNull permitem especificar que as pré-condições nas variáveis podem não
corresponder às anotações anuláveis nessas variáveis. Eles fornecem mais detalhes sobre as características de sua
API. Essas informações adicionais ajudam os chamadores a usar sua API corretamente. Lembre-se de especificar
as pré-condições usando os seguintes atributos:
AllowNull: um argumento de entrada não anulável pode ser nulo.
DisallowNull: um argumento de entrada anulável nunca deve ser nulo.
Você provavelmente escreveu um método como este para retornar null quando o nome procurado não foi
encontrado. O null indica claramente que o registro não foi encontrado. Neste exemplo, você provavelmente
alteraria o tipo de retorno de Customer para Customer? . Declarar o valor de retorno como um tipo de referência
anulável especifica claramente a intenção dessa API.
Por motivos abordados em definições genéricas e nulidade que a técnica não funciona com métodos genéricos.
Você pode ter um método genérico que segue um padrão semelhante:
Você não pode especificar que o valor de retorno é T? . O método retorna null quando o item procurado não é
encontrado. Como não é possível declarar um tipo de retorno T? , você adiciona a anotação MaybeNull ao retorno
do método:
[return: MaybeNull]
public T Find<T>(IEnumerable<T> sequence, Func<T, bool> match)
O código anterior informa aos chamadores que o contrato implica em um tipo não anulável, mas o valor de
retorno pode ser, na verdade, nulo. Use o atributo MaybeNull quando sua API deve ser um tipo não anulável,
normalmente um parâmetro de tipo genérico, mas pode haver instâncias em que null seriam retornados.
Você também pode especificar que um valor de retorno ou um argumento out ou ref não seja nulo, embora o
tipo seja um tipo anulável. Considere um método que garanta que uma matriz seja grande o suficiente para conter
vários elementos. Se o argumento de entrada não tiver capacidade, a rotina alocaria uma nova matriz e copiaria
todos os elementos existentes nela. Se o argumento de entrada for null , a rotina alocaria novo armazenamento.
Se houver capacidade suficiente, a rotina não fará nada:
Depois de habilitar tipos de referência NULL, você deseja garantir que o código anterior seja compilado sem
avisos. Quando o método retorna, o argumento storage é garantido como não nulo. No entanto, é aceitável
chamar EnsureCapacity com uma referência nula. Você pode fazer storage um tipo de referência anulável e
adicionar o NotNull pós-condição à declaração de parâmetro:
O código anterior expressa o contrato existente muito claramente: os chamadores podem passar uma variável com
o valor null , mas é garantido que o valor de retorno nunca seja nulo. O atributo NotNull é mais útil para
argumentos ref e out em que null pode ser passado como um argumento, mas esse argumento é garantido
como não nulo quando o método retorna.
Você especifica condições de erro incondicionais usando os seguintes atributos:
MaybeNull: um valor de retorno não anulável pode ser nulo.
Não nulo: um valor de retorno anulável nunca será nulo.
Isso informa ao compilador que qualquer código em que o valor de retorno é false não precisa ser marcado
como nulo. A adição do atributo informa a análise estática do compilador que IsNullOrEmpty executa a verificação
nula necessária: quando ela retorna false , o argumento de entrada não é null .
string? userInput = GetUserInput();
if (!string.IsNullOrEmpty(userInput))
{
int messageLength = userInput.Length; // no null check needed.
}
// null check needed on userInput here.
O método de String.IsNullOrEmpty(String) será anotado conforme mostrado acima para o .NET Core 3,0. Você
pode ter métodos semelhantes em sua base de código que verificam o estado dos objetos em busca de valores
nulos. O compilador não reconhecerá os métodos de verificação NULL personalizados e você precisará adicionar
as anotações por conta própria. Quando você adiciona o atributo, a análise estática do compilador sabe quando a
variável testada foi marcada como nula.
Outro uso para esses atributos é o padrão de Try* . As condições para ref e out variáveis são comunicadas por
meio do valor de retorno. Considere este método mostrado anteriormente:
O método anterior segue um idioma .NET típico: o valor de retorno indica se message foi definido como o valor
encontrado ou, se nenhuma mensagem for encontrada, para o valor padrão. Se o método retornar true , o valor
de message não será nulo; caso contrário, o método define message como NULL.
Você pode comunicar esse idioma usando o atributo NotNullWhen . Quando você atualiza a assinatura para tipos de
referência anuláveis, faça message um string? e adicione um atributo:
No exemplo anterior, o valor de message é conhecido como não nulo quando TryGetMessage retorna true . Você
deve anotar métodos semelhantes em sua base de código da mesma maneira: os argumentos podem ser null dos
e ser conhecidos como não nulos quando o método retornar true .
Há um atributo final que você também pode precisar. Às vezes, o estado nulo de um valor de retorno depende do
estado nulo de um ou mais argumentos de entrada. Esses métodos retornarão um valor não nulo sempre que
determinados argumentos de entrada não forem null . Para anotar corretamente esses métodos, use o atributo
NotNullIfNotNull . Considere o seguinte método:
Se o argumento url não for nulo, a saída não será null . Depois que as referências anuláveis estiverem
habilitadas, essa assinatura funcionará corretamente, desde que sua API nunca aceite uma entrada nula. No
entanto, se a entrada puder ser nula, o valor de retorno também poderá ser nulo. Portanto, você pode alterar a
assinatura para o código a seguir:
Isso também funciona, mas, muitas vezes, forçará os chamadores a implementarem verificações de null
adicionais. O contrato é que o valor de retorno seria null somente quando o argumento de entrada url é null .
Para expressar esse contrato, você anotaria esse método, conforme mostrado no código a seguir:
[return: NotNullIfNotNull("url")]
string? GetTopLevelDomainFromFullUrl(string? url);
O valor de retorno e o argumento foram anotados com o ? indicando que pode ser null . O atributo esclarece
ainda mais que o valor de retorno não será nulo quando o argumento url não for null .
Você especifica condições recondicionais usando estes atributos:
MaybeNullWhen: um argumento de entrada não anulável pode ser nulo quando o método retorna o valor de
bool especificado.
NotNullWhen: um argumento de entrada anulável não será nulo quando o método retornar o valor de bool
especificado.
NotNullIfNotNull: um valor de retorno não será nulo se o argumento de entrada para o parâmetro especificado
não for nulo.
Isso não significa que você não pode usar um tipo anulável (tipo de valor ou tipo de referência) como o argumento
de tipo para um tipo genérico fechado. Tanto List<string?> quanto List<int?> são instanciações válidas de
List<T> .
O que significa é que você não pode usar T? em uma classe genérica ou declaração de método sem restrições.
Por exemplo, Enumerable.FirstOrDefault<TSource>(IEnumerable<TSource>) não será alterado para retornar T? .
Você pode superar essa limitação adicionando a restrição struct ou class . Com qualquer uma dessas restrições,
o compilador sabe como gerar código para T e T? .
Talvez você queira restringir os tipos usados para que um argumento de tipo genérico seja de tipos não anuláveis.
Você pode fazer isso adicionando a restrição de notnull nesse argumento de tipo. Quando essa restrição é
aplicada, o argumento de tipo não deve ser um tipo anulável.
Conclusões
A adição de tipos de referência anuláveis fornece um vocabulário inicial para descrever suas expectativas de APIs
para variáveis que podem ser null . Os atributos adicionais fornecem um vocabulário mais rico para descrever o
estado nulo de variáveis como pré-condições e condições. Esses atributos descrevem mais claramente suas
expectativas e fornecem uma experiência melhor para os desenvolvedores que usam suas APIs.
Conforme você atualiza as bibliotecas para um contexto anulável, adicione esses atributos para orientar os
usuários de suas APIs para o uso correto. Esses atributos ajudam você a descrever totalmente o estado nulo dos
argumentos de entrada e valores de retorno:
AllowNull: um argumento de entrada não anulável pode ser nulo.
DisallowNull: um argumento de entrada anulável nunca deve ser nulo.
MaybeNull: um valor de retorno não anulável pode ser nulo.
Não nulo: um valor de retorno anulável nunca será nulo.
MaybeNullWhen: um argumento de entrada não anulável pode ser nulo quando o método retorna o valor de
bool especificado.
NotNullWhen: um argumento de entrada anulável não será nulo quando o método retornar o valor de bool
especificado.
NotNullIfNotNull: um valor de retorno não será nulo se o argumento de entrada para o parâmetro especificado
não for nulo.
Namespaces (Guia de Programação em C#)
23/10/2019 • 2 minutes to read • Edit Online
Os namespaces são usados intensamente em programações de C# de duas maneiras. Em primeiro lugar, o .NET
Framework usa namespaces para organizar suas muitas classes, da seguinte maneira:
System.Console.WriteLine("Hello World!");
System é um namespace e Console é uma classe nesse namespace. A palavra-chave using pode ser usada para
que o nome completo não seja necessário, como no exemplo a seguir:
using System;
Console.WriteLine("Hello");
Console.WriteLine("World!");
namespace SampleNamespace
{
class SampleClass
{
public void SampleMethod()
{
System.Console.WriteLine(
"SampleMethod inside SampleNamespace");
}
}
}
Especificação da linguagem C#
Para saber mais, confira a seção Namespaces da Especificação da linguagem C#.
Consulte também
Guia de Programação em C#
Usando namespaces
Como: usar o My Namespace
Nomes de identificadores
Diretiva using
:: ??
Tipos, variáveis e valores
29/11/2019 • 12 minutes to read • Edit Online
C# é uma linguagem fortemente tipada. Todas as variáveis e constantes têm um tipo, assim como cada expressão
que é avaliada como um valor. Cada assinatura de método especifica um tipo para cada parâmetro de entrada e
para o valor retornado. A biblioteca de classes .NET Framework define um conjunto de tipos numéricos internos,
bem como tipos mais complexos que representam uma ampla variedade de constructos lógicos, como o sistema
de arquivos, as conexões de rede, as coleções e as matrizes de objetos e as datas. Um programa em C# típico usa
tipos da biblioteca de classes, bem como tipos definidos pelo usuário que modelam os conceitos que são
específicos para o domínio do problema do programa.
As informações armazenadas em um tipo podem incluir o seguinte:
O espaço de armazenamento que uma variável do tipo requer.
Os valores mínimo e máximo que ele pode representar.
Os membros (métodos, campos, eventos e etc.) que ele contém.
O tipo base do qual ele herda.
O local no qual a memória para as variáveis será alocada em tempo de execução.
Os tipos de operações que são permitidos.
O compilador usa as informações de tipo para garantir que todas as operações que são realizadas em seu código
sejam fortemente tipadas. Por exemplo, se você declarar uma variável do tipo int, o compilador permitirá que você
use a variável nas operações de adição e subtração. Se você tentar executar as mesmas operações em uma variável
do tipo bool, o compilador gerará um erro, como mostrado no exemplo a seguir:
int a = 5;
int b = a + 2; //OK
// Error. Operator '+' cannot be applied to operands of type 'int' and 'bool'.
int c = a + test;
NOTE
Desenvolvedores de C e C++, observem que, em C#, bool não é conversível em int.
O compilador insere as informações de tipo no arquivo executável como metadados. O CLR (Common Language
Runtime) usa metadados em tempo de execução para garantir mais segurança de tipos quando aloca e recupera a
memória.
Os tipos de parâmetros de método e valores de retorno são especificados na assinatura do método. A assinatura a
seguir mostra um método que requer um int como um argumento de entrada e retorna uma cadeia de caracteres:
Depois que uma variável é declarada, ela não pode ser declarada novamente com um novo tipo e não pode ter um
valor atribuído que não seja compatível com seu tipo declarado. Por exemplo, você não pode declarar um int e, em
seguida, atribuir a ele um valor booliano de true . No entanto, os valores podem ser convertidos em outros tipos,
por exemplo, quando são passados como argumentos de método ou atribuídos a novas variáveis. Uma conversão
de tipo que não causa a perda de dados é executada automaticamente pelo compilador. Uma conversão que pode
causar perda de dados requer um cast no código-fonte.
Para obter mais informações, consulte Conversões cast e conversões de tipo.
Tipos internos
O C# fornece um conjunto padrão de tipos numéricos internos para representar números inteiros, valores de
ponto flutuante, expressões boolianas, caracteres de texto, valores decimais e outros tipos de dados. Também há
tipos internos de cadeias de caracteres de objetos. Eles estão disponíveis para uso em qualquer programa em
C#. Para obter mais informações sobre os tipos internos, consulte Tabelas de referência de tipos internos.
Tipos personalizados
Você usa os constructos struct, classe, interface e enum para criar seus próprios tipos personalizados. A biblioteca
de classes .NET Framework em si é uma coleção de tipos personalizados fornecidos pela Microsoft que você pode
usar em seus próprios aplicativos. Por padrão, os tipos usados com mais frequência na biblioteca de classes estão
disponíveis em qualquer programa em C#. Outros ficam disponíveis somente quando você adiciona explicitamente
uma referência de projeto ao assembly no qual eles estão definidos. Após o compilador ter uma referência ao
assembly, você pode declarar variáveis (e constantes) dos tipos declarados no assembly no código-fonte.
Tipos genéricos
Um tipo pode ser declarado com um ou mais parâmetros de tipo que servem como um espaço reservado para o
tipo real (o tipo concreto) que o código do cliente fornecerá ao criar uma instância do tipo. Esses tipos são
chamados de tipos genéricos. Por exemplo, o tipo de .NET Framework List<T> tem um parâmetro de tipo que, por
convenção, recebe o nome t. Ao criar uma instância do tipo, você especifica o tipo dos objetos que a lista conterá,
por exemplo, Cadeia de caracteres:
O uso do parâmetro de tipo possibilita a reutilização da mesma classe para conter qualquer tipo de elemento sem
precisar converter cada elemento em objeto. As classes de coleção genéricas são chamadas de coleções fortemente
tipadas porque o compilador sabe o tipo específico dos elementos da coleção e pode gerar um erro em tempo de
compilação se, por exemplo, você tentar adicionar um inteiro ao objeto strings no exemplo anterior. Para obter
mais informações, consulte Genéricos.
Consulte também
Structs
Classes
Classes (Guia de Programação em C#)
23/10/2019 • 9 minutes to read • Edit Online
Tipos de referência
Um tipo que é definido como uma classe é um tipo de referência. No tempo de execução, quando você declara uma
variável de um tipo de referência, a variável contém o valor null até que você crie explicitamente uma instância da
classe usando o operador new ou atribua a ela um objeto de um tipo compatível que foi criado em outro lugar,
conforme mostrado no exemplo a seguir:
//Declaring another object of the same type, assigning it the value of the first object.
MyClass mc2 = mc;
Quando o objeto é criado, memória suficiente é alocada no heap gerenciado para o objeto específico, e a variável
contém apenas uma referência para o local do objeto. Os tipos no heap gerenciado requerem sobrecarga quando
são alocados e quando são recuperados pela funcionalidade de gerenciamento automático de memória do CLR,
que é conhecida como coleta de lixo. No entanto, a coleta de lixo também é altamente otimizada e, na maioria dos
cenários, não cria um problema de desempenho. Para obter mais informações sobre a coleta de lixo, consulte
Gerenciamento automático de memória e coleta de lixo.
Declarando Classes
As classes são declaradas usando a palavra-chave class, seguida por um identificador exclusivo, conforme
mostrado no exemplo a seguir:
A palavra-chave class é precedida pelo nível de acesso. Como público é usado nesse caso, qualquer pessoa pode
criar instâncias dessa classe. O nome da classe segue a palavra-chave class . O nome da classe deve ser um nome
do identificador válido em C#. O restante da definição é o corpo da classe, em que o comportamento e os dados
são definidos. Campos, propriedades, métodos e eventos em uma classe são coletivamente denominados de
membros de classe.
Criando objetos
Embora eles sejam usados algumas vezes de maneira intercambiável, uma classe e um objeto são coisas diferentes.
Uma classe define um tipo de objeto, mas não é um objeto em si. Um objeto é uma entidade concreta com base em
uma classe e, às vezes, é conhecido como uma instância de uma classe.
Os objetos podem ser criados usando a palavra-chave new, seguida pelo nome da classe na qual ele se baseará,
dessa maneira:
Customer object1 = new Customer();
Quando uma instância de uma classe é criada, uma referência ao objeto é passada de volta para o programador. No
exemplo anterior, object1 é uma referência a um objeto que é baseado em Customer . Esta referência refere-se ao
novo objeto, mas não contém os dados de objeto. Na verdade, você pode criar uma referência de objeto sem criar
um objeto:
Customer object2;
Não recomendamos a criação de referências de objeto como essa, que não faz referência a um objeto, porque
tentar acessar um objeto por meio de uma referência desse tipo falhará em tempo de execução. Entretanto, essa
referência pode ser feita para se referir a um objeto, criando um novo objeto ou atribuindo-a a um objeto existente,
como abaixo:
Esse código cria duas referências de objeto que fazem referência ao mesmo objeto. Portanto, qualquer alteração no
objeto feita por meio de object3 será refletida no usos posteriores de object4 . Como os objetos que são
baseados em classes são referenciados por referência, as classes são conhecidas como tipos de referência.
Herança de classe
As classes dão suporte completo à herança, uma característica fundamental da programação orientada a objetos.
Ao criar uma classe, você pode herdar de outra interface ou classe que não está definida como selada, e outras
classes podem herdar de sua classe e substituir seus métodos virtuais.
A herança é realizada usando uma derivação, o que significa que uma classe é declarada usando uma classe base,
da qual ela herda o comportamento e os dados. Uma classe base é especificada ao acrescentar dois-pontos e o
nome de classe base depois do nome de classe derivada, dessa maneira:
Quando uma classe declara uma classe base, ela herda todos os membros da classe base, exceto os construtores.
Para obter mais informações, consulte Herança.
Ao contrário do C++, uma classe no C# só pode herdar diretamente de uma classe base. No entanto, como uma
classe base pode herdar de outra classe, uma classe pode herdar indiretamente várias classes base. Além disso,
uma classe pode implementar diretamente mais de uma interface. Para obter mais informações, consulte Interfaces.
Uma classe pode ser declarada abstract. Uma classe abstrata contém métodos abstratos que têm uma definição de
assinatura, mas não têm implementação. As classes abstratas não podem ser instanciadas. Elas só podem ser
usadas por meio de classes derivadas que implementam os métodos abstratos. Por outro lado, uma classe lacrada
não permite que outras classes sejam derivadas dela. Para obter mais informações, consulte Classes e Membros de
Classes Abstratos e Lacrados.
As definições de classe podem ser divididas entre arquivos de origem diferentes. Para obter mais informações,
consulte Classes e métodos parciais.
Exemplo
No exemplo a seguir, é definida uma classe pública que contém uma propriedade autoimplementada, um método e
um método especial chamado construtor. Para obter mais informações, consulte os tópicos Propriedades, Métodos,
e Construtores. As instâncias da classe são então instanciadas com a palavra-chave new .
using System;
Especificação da Linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a fonte
definitiva para a sintaxe e o uso de C#.
Consulte também
Guia de Programação em C#
Programação Orientada a Objeto
Polimorfismo
Nomes de identificadores
Membros
Métodos
Construtores
Finalizadores
Objetos
Structs
08/11/2019 • 5 minutes to read • Edit Online
Um struct é um tipo de valor. Quando um struct é criado, a variável à qual o struct está atribuído contém os dados
reais do struct. Quando o struct é atribuído a uma nova variável, ele é copiado. A nova variável e a variável
original, portanto, contêm duas cópias separadas dos mesmos dados. As alterações feitas em uma cópia não
afetam a outra cópia.
As variáveis de tipo de valor contêm diretamente seus valores, o que significa que a memória é alocada embutida
em qualquer contexto em que a variável é declarada. Não há nenhuma alocação de heap separada ou sobrecarga
de coleta de lixo para variáveis do tipo de valor.
Há duas categorias de tipos de valor: struct e enum.
Os tipos numéricos internos são structs e têm propriedades e métodos que você pode acessar:
Mas você declara e atribui valores a eles como se fossem tipos de não agregação simples:
Tipos de valor são lacrados, o que significa, por exemplo, que você não pode derivar um tipo de Int32 e não pode
definir um struct para herdar de qualquer struct ou classe definida pelo usuário, porque um struct apenas pode
herdar de ValueType. No entanto, um struct pode implementar uma ou mais interfaces. Você pode converter um
tipo de struct em um tipo de interface. Isso faz com que uma operação de conversão boxing encapsule o struct
dentro de um objeto de tipo de referência no heap gerenciado. Operações de conversão boxing ocorrem quando
você passa um tipo de valor para um método que usa um Object como parâmetro de entrada. Para obter mais
informações, consulte Boxing e unboxing.
Você usa a palavra-chave struct para criar seus próprios tipos de valor personalizados. Normalmente, um struct é
usado como um contêiner para um pequeno conjunto de variáveis relacionadas, conforme mostrado no exemplo
a seguir:
Para obter mais informações sobre os tipos de valor no .NET Framework, consulte Common Type System.
Na maioria das vezes, os structs compartilham a mesma sintaxe das classes, embora os structs sejam mais
limitados que as classes:
Dentro de uma declaração de struct, os campos não podem ser inicializados, a menos que sejam declarados
como const ou static .
Um struct não pode declarar um construtor sem padrão (um construtor sem parâmetros) ou um finalizador.
Os structs são copiados na atribuição. Quando um struct recebe uma nova variável, todos os dados são
copiados e qualquer modificação na nova cópia não altera os dados da cópia original. É importante se
lembrar disso ao trabalhar com coleções de tipos de valor como Dictionary<string, myStruct>.
Structs são tipos de valor e classes são tipos de referência.
Diferentemente das classes, os structs podem ser instanciados sem usar um operador new .
Os structs podem declarar construtores que têm parâmetros.
Um struct não pode herdar de outra estrutura ou classe e ele não pode ser a base de uma classe. Todos os
structs herdam diretamente do ValueType, que herda do Object.
Um struct pode implementar interfaces.
Consulte também
Classes
Tipos Básicos
Tipos de tupla do C#
30/10/2019 • 36 minutes to read • Edit Online
As tuplas do C# são tipos que você define usando uma sintaxe leve. As vantagens incluem sintaxe mais
simples, regras para conversões baseadas em números (conhecidas como cardinalidade) e em tipos de
elementos, além de regras compatíveis para cópias, testes de igualdade e atribuições. Em contrapartida, as
tuplas não oferecem suporte a algumas das expressões orientadas a objeto associadas à herança. Você pode
obter uma visão geral na seção sobre tuplas no artigo Novidades no C# 7.0.
Neste artigo, você aprenderá as regras de linguagem que regem as tuplas no C# 7.0 e em versões
posteriores, além de diferentes maneiras de usá-las e as diretrizes iniciais para trabalhar com tuplas.
NOTE
Os novos recursos de tuplas exigem os tipos ValueTuple. Você deve adicionar o pacote NuGet System.ValueTuple
para usá-lo em plataformas que não incluem os tipos.
Isso é semelhante a outros recursos de linguagem que dependem de tipos entregues no framework. Os exemplos
incluem async e await que dependem da interface INotifyCompletion , além do LINQ que depende de
IEnumerable<T> . No entanto, o mecanismo de entrega está mudando conforme o .NET se torna mais independente
de plataforma. O .NET Framework pode não ser enviados sempre na mesma cadência que o compilador de linguagem.
Quando novos recursos de linguagem dependerem de novos tipos, esses tipos estarão disponíveis como pacotes do
NuGet quando os recursos de linguagem forem enviados. Conforme esses novos tipos são adicionados à API padrão
do .NET e fornecidos como parte do framework, o requisito de pacote do NuGet será removido.
Vamos começar com os motivos para adicionar o novo suporte de tupla. Métodos retornam um único objeto.
Tuplas permitem que você empacote vários valores nesse único objeto mais facilmente.
O .NET Framework já tem classes Tuple genéricas. Essas classes, no entanto, têm duas limitações
importantes. Por exemplo, as classes Tuple nomearam suas propriedades como Item1 , Item2 e assim por
diante. Esses nomes não carregam informações semânticas. O uso desses tipos Tuple não permite
comunicar o significado de cada uma das propriedades. Os novos recursos de linguagem permitem que você
declare e use nomes semanticamente significativos para os elementos em uma tupla.
As classes de Tuple causam mais problemas de desempenho porque elas são tipos de referência. O uso de
um dos tipos Tuple significa alocar objetos. Em afunilamentos, alocar muitos objetos pequenos pode ter um
impacto mensurável no desempenho do aplicativo. Portanto, o suporte de linguagem para tuplas aproveita os
novos structs ValueTuple .
Para evitar essas deficiências, você pode criar uma class ou um struct para carregar vários elementos.
Infelizmente, isso significa mais trabalho para você e obscurece a intenção do design. Fazer uma struct ou
class significa que você está definindo um tipo com os dados e comportamento. Muitas vezes, você
simplesmente deseja armazenar diversos valores em um único objeto.
Os recursos de linguagem e os structs genéricos ValueTuple aplicam a regra de que você não pode adicionar
nenhum comportamento (método) a esses tipos de tupla. Todos os tipos ValueTuple são structs mutáveis.
Cada campo membro é um campo público. Isso os torna muito simples. No entanto, isso significa que as
tuplas não devem ser usadas quando a imutabilidade é importante.
As tuplas são contêineres de dados mais simples e mais flexíveis do que os tipos class e struct . Vamos
explorar essas diferenças.
Tuplas nomeadas e sem nome
O struct ValueTuple tem campos nomeados Item1 , Item2 , Item3 e assim por diante, semelhante às
propriedades definidas nos tipos de Tuple existentes. Esses nomes são os únicos nomes que você pode usar
para tuplas sem nome. Quando você não fornece nomes de campo alternativos para uma tupla, cria uma
tupla sem nome:
A tupla no exemplo anterior foi inicializada usando constantes literais e não terá nomes de elemento criados
usando as projeções de nome de campo de tupla no C# 7.1.
No entanto, quando você inicializa uma tupla, pode usar novos recursos de linguagem que fornecem nomes
melhores para cada campo. Fazer isso cria uma tupla nomeada. As tuplas nomeadas ainda têm elementos
chamados Item1 , Item2 , Item3 e assim por diante. Mas também têm sinônimos para qualquer um desses
elementos que você tenha nomeado. Você cria uma tupla nomeada especificando os nomes de cada
elemento. Uma maneira é especificar os nomes como parte da inicialização da tupla:
Esses sinônimos são manipulados pelo compilador e pela linguagem para que você possa usar as tuplas
nomeadas de forma eficaz. Os editores e IDEs podem ler esses nomes semânticos usando APIs Roslyn. É
possível fazer referência aos elementos de uma tupla nomeada com nomes semânticos em qualquer lugar no
mesmo assembly. O compilador substitui os nomes que você definiu com equivalentes Item* ao gerar a
saída compilada. A MSIL (Microsoft Intermediate Language) compilada não inclui os nomes que você
atribuiu a esses elementos.
Começando com o C# 7.1, os nomes de campo para uma tupla podem ser fornecidos por meio das variáveis
usadas para inicializar a tupla. Isso é conhecido como inicializadores de projeção de tupla . O código a
seguir cria uma tupla denominada accumulation com elementos count (um inteiro) e sum (um duplo).
O compilador deve comunicar esses nomes que você criou para tuplas retornadas de métodos públicos ou
propriedades. Nesses casos, o compilador adiciona um atributo TupleElementNamesAttribute no método.
Esse atributo contém uma propriedade de lista TransformNames que inclui os nomes dados a cada um desses
elementos na tupla.
NOTE
Ferramentas de desenvolvimento, como o Visual Studio, também leem esses metadados e fornecem IntelliSense e
outros recursos usando os nomes de campo de metadados.
É importante entender esses conceitos básicos subjacentes das novas tuplas e do tipo ValueTuple para
entender as regras para atribuir tuplas nomeadas entre si.
var localVariableOne = 5;
var localVariableTwo = "some text";
Para qualquer campo em que um nome explícito não for fornecido, será projetado um nome implícito
aplicável. Não há nenhum requisito para fornecer nomes semânticos, explícita ou implicitamente. O
inicializador a seguir terá nomes de campo Item1 , cujo valor é 42 e stringContent , cujo valor é "A resposta
para tudo":
Há duas condições nas quais os possíveis nomes de campos não são projetados no campo da tupla:
1. Quando o possível nome é um nome de tupla reservado. Os exemplos incluem Item3 , ToString ou Rest
.
2. Quando o possível nome é uma duplicata de outro nome de campo de tupla, seja explícito ou implícito.
Essas condições evitam a ambiguidade. Esses nomes causariam ambiguidade se fossem usados como nomes
de campo em uma tupla. Nenhuma dessas condições causa erros de tempo de compilação. Em vez disso, os
elementos sem nomes projetados não terão nomes semânticos projetados para eles. Os exemplos a seguir
demonstram essas condições:
Essas situações não causam erros de compilador porque essa seria uma alteração significativa nos códigos
escritos com C# 7.0, em que as projeções de nome de campo de tupla não estavam disponíveis.
Igualdade e tuplas
Começando com o C# 7.3, os tipos de tupla oferecem suporte aos operadores == e != . Esses operadores
comparam cada membro do argumento da esquerda com cada membro do argumento da direita na ordem.
Essas comparações são de curto-circuito. Elas interromperão a avaliação de membros assim que um par não
for igual. O código a seguir exemplifica o uso de == , mas todas as regras de comparação se aplicam a != . O
exemplo de código a seguir mostra uma comparação de igualdade de dois pares de inteiros:
Há várias regras que tornam os testes de igualdade de tuplas mais convenientes. A igualdade de tupla executa
conversões lifted se uma das tuplas for uma tupla que permite valor nulo, conforme mostrado no código a
seguir:
A igualdade de tupla também executa conversões implícitas em cada membro de ambas as tuplas. Esses
incluem conversões lifted, conversões widening ou outras conversões implícitas. Os exemplos a seguir
mostram que um inteiro de uma tupla 2 pode ser comparado a um longo de tupla 2 devido à conversão
implícita do inteiro para um longo:
// lifted conversions
var left = (a: 5, b: 10);
(int? a, int? b) nullableMembers = (5, 10);
Console.WriteLine(left == nullableMembers); // Also true
Os nomes dos membros da tupla não participam em testes de igualdade. No entanto, se um dos operandos
for uma tupla literal com nomes explícitos, o compilador gera o aviso CS8383 caso os nomes não
correspondam aos nomes do outro operando. No caso em que ambos os operandos são literais de tupla, o
aviso é no operando à direita, conforme mostrado no exemplo a seguir:
Por fim, as tuplas podem conter tuplas aninhadas. A igualdade de tupla compara a "forma" de cada operando
por meio de tuplas aninhadas, conforme mostrado no exemplo a seguir:
É um erro de tempo de compilação comparar duas tuplas por igualdade (ou desigualdade) quando elas têm
formas diferentes. O compilador não tentará nenhuma desconstrução das tuplas aninhadas para compará-las.
Atribuição e tuplas
A linguagem oferece suporte à atribuição entre tipos de tuplas que têm o mesmo número de elementos, em
que cada elemento do lado direito pode ser convertido implicitamente em seu elemento correspondente do
lado esquerdo. Outras conversões não são consideradas para atribuições. É um erro de tempo de compilação
atribuir uma tupla a outra quando elas têm formas diferentes. O compilador não tentará nenhuma
desconstrução das tuplas aninhadas para atribuí-las. Vamos examinar os tipos de atribuições que são
permitidos entre tipos de tupla.
Considere estas variáveis usadas nos exemplos a seguir:
As primeiras duas variáveis, unnamed e anonymous , não têm nomes semânticos fornecidos para os elementos.
Os nomes de campo são Item1 e Item2 . As duas últimas variáveis, named e differentName , têm nomes
semânticos fornecidos para os elementos. Estas duas tuplas têm nomes diferentes para os elementos.
Todas essas quatro tuplas têm o mesmo número de elementos (chamados de "cardinalidade") e os tipos
desses elementos são idênticos. Portanto, todas essas atribuições funcionam:
unnamed = named;
named = unnamed;
// 'named' still has fields that can be referred to
// as 'answer', and 'message':
Console.WriteLine($"{named.Answer}, {named.Message}");
// unnamed to unnamed:
anonymous = unnamed;
// named tuples.
named = differentNamed;
// The field names are not assigned. 'named' still has
// fields that can be referred to as 'answer' and 'message':
Console.WriteLine($"{named.Answer}, {named.Message}");
Observe que os nomes das tuplas não são atribuídos. Os valores dos elementos são atribuídos na ordem dos
elementos na tupla.
As tuplas com tipos ou números de elementos diferentes não são atribuíveis:
NOTE
Esses exemplos calculam o desvio padrão de exemplo não corrigido. A fórmula do desvio padrão de exemplo corrigida
dividiria a soma das diferenças da média ao quadrado por (N-1) em vez de N, como o método de extensão Average
faz. Consulte um texto sobre estatísticas para obter mais detalhes sobre as diferenças entre essas fórmulas para desvio
padrão.
O código anterior segue a fórmula típica para o desvio padrão. Ele produz a resposta correta, mas é uma
implementação ineficiente. Esse método enumera a sequência de duas vezes: uma vez para produzir a média
e uma vez para produzir a média do quadrado da diferença da média. (Lembre-se de que consultas LINQ são
avaliadas lentamente, então o cálculo das diferenças da média e a média dessas diferenças compõem apenas
uma enumeração.)
Há uma alternativa fórmula que calcula o desvio padrão usando apenas uma enumeração da sequência. Esse
cálculo produz dois valores conforme enumera a sequência: a soma de todos os itens na sequência e a soma
de cada valor ao quadrado:
Essa versão enumera a sequência exatamente uma vez. Mas não é um código reutilizável. Conforme você
continua a trabalhar, descobrirá que muitos cálculos estatísticos diferentes usam o número de itens na
sequência, a soma da sequência e a soma dos quadrados da sequência. Vamos refatorar esse método e
escrever um método utilitário que produz todos esses três valores. Todos os três valores podem retornar
como uma tupla.
Vamos atualizar esse método para que os três valores calculados durante a enumeração sejam armazenados
em uma tupla. Isso cria essa versão:
O suporte à refatoração do Visual Studio facilita a extração da funcionalidade para as estatísticas principais
em um método privado. Isso fornece a você um método private static que retorna o tipo de tupla com os
três valores de Sum , SumOfSquares e Count :
return computation;
}
A linguagem permite algumas opções adicionais que podem ser usadas se você desejar fazer algumas
edições rápidas manualmente. Primeiro, você pode usar a declaração var para inicializar o resultado da tupla
da chamada do método ComputeSumAndSumOfSquares . Você também pode criar três variáveis discretas dentro
do método ComputeSumAndSumOfSquares . A versão final é mostrada no código a seguir:
public static double StandardDeviation(IEnumerable<double> sequence)
{
var computation = ComputeSumAndSumOfSquares(sequence);
Esta versão final pode ser usada para qualquer método que precise desses três valores ou qualquer
subconjunto deles.
A linguagem dá suporte a outras opções no gerenciamento dos nomes dos elementos nesses métodos de
retorno de tupla.
Você pode remover os nomes de campo da declaração de valor retornado e retornar uma tupla sem nome:
Os campos nesta tupla são nomeados Item1 , Item2 e Item3 . É recomendável que você forneça nomes
semânticos para os elementos de tuplas retornados dos métodos.
Outra linguagem em que as tuplas podem ser úteis é quando você estiver criando consultas LINQ. O
resultado final projetado geralmente contém algumas, mas não todas, propriedades dos objetos que estão
sendo selecionados.
Tradicionalmente, você poderia projetar os resultados da consulta em uma sequência de objetos que eram um
tipo anônimo. Isso apresentava muitas limitações, principalmente porque os tipos anônimos não poderiam
ser nomeados de forma conveniente no tipo de retorno de um método. Alternativas usando object ou
dynamic como o tipo de resultado acompanham os custos de desempenho significativos.
Retornar uma sequência de um tipo de tupla é fácil e os nomes e tipos desses elementos estão disponíveis no
tempo de compilação e por meio de ferramentas de IDE. Por exemplo, imagine um aplicativo de tarefas. Você
pode definir uma classe semelhante à seguinte para representar uma única entrada na lista de tarefas:
Seus aplicativos móveis podem dar suporte a um formato compacto dos itens de tarefas atuais que exibem
apenas o título. Essa consulta LINQ faria uma projeção que inclui somente a ID e o título. Um método que
retorna uma sequência de tuplas expressa bem esse design:
NOTE
No C# 7.1, as projeções de tupla permitem criar tuplas nomeadas usando elementos, de maneira semelhante à
nomeação de propriedade nos tipos anônimos. No código acima, a instrução select na projeção de consulta cria uma
tupla que tem elementos ID e Title .
A tupla nomeada pode ser parte da assinatura. Ela permite que o compilador e as ferramentas de IDE
forneçam verificações estáticas de que você está usando o resultado corretamente. A tupla nomeada também
contém as informações de tipo estático, portanto, não há necessidade de usar recursos caros de tempo de
execução como reflexão ou associação dinâmica para trabalhar com os resultados.
Desconstrução
Você pode descompactar todos os itens em uma tupla desconstruindo a tupla retornada por um método. Há
três abordagens diferentes para desconstruir tuplas. Primeiro, você pode declarar explicitamente o tipo de
cada campo dentro de parênteses para criar variáveis discretas para cada um dos elementos na tupla:
Você também pode declarar variáveis de tipo implícito para cada campo em uma tupla usando a palavra-
chave var fora dos parênteses:
public static double StandardDeviation(IEnumerable<double> sequence)
{
var (sum, sumOfSquares, count) = ComputeSumAndSumOfSquares(sequence);
Também é válido usar a palavra-chave var com qualquer uma ou todas as declarações de variável dentro
dos parênteses.
Você não poderá usar um tipo específico fora dos parênteses, mesmo se todos os campos na tupla tiverem o
mesmo tipo.
Você também pode desconstruir tuplas com declarações existentes:
WARNING
Você não pode combinar as declarações existentes com as declarações dentro dos parênteses. Por exemplo, o seguinte
não é permitido: (var x, y) = MyMethod(); . Isso gera o erro CS8184 porque x está declarado dentro dos parênteses
e y já foi declarado em outro lugar.
O método deconstruct permite a atribuição de um Person para duas cadeias de caracteres, representando as
propriedades FirstName e LastName :
Você pode habilitar a desconstrução mesmo para os tipos que você não criou. O método Deconstruct pode
ser um método de extensão que retira do pacote os membros de dados acessíveis de um objeto. O exemplo a
seguir mostra um tipo Student , derivado do tipo Person e um método de extensão que desconstrói um
Student em três variáveis, representando a FirstName , a LastName e a GPA :
Um objeto Student agora tem dois métodos Deconstruct acessíveis: o método de extensão declarado para
tipos Student e o membro do tipo Person . Ambos estão no escopo e isso permite que um Student seja
desconstruído em duas ou três variáveis. Se você atribuir um aluno a três variáveis, o nome, o sobrenome e a
GPA serão retornados. Se você atribuir um aluno a duas variáveis, apenas o nome e o sobrenome serão
retornados.
Os operadores de desconstrução não participam do teste de igualdade. O exemplo a seguir gera o erro do
compilador CS0019:
O método Deconstruct pôde converter o objeto Person p em uma tupla que contém duas cadeias de
caracteres, mas não é aplicável no contexto de testes de igualdade.
Console.WriteLine($"{pair.num}: {pair.place}");
/*
* Output:
* 345: Second
*/
Como alternativa, você pode usar uma tupla sem nome e fazer referência a seus campos como Item1 e
Item2 :
Conclusão
O novo suporte de linguagem e biblioteca para tuplas nomeadas torna muito mais fácil trabalhar com
designs que usam estruturas de dados que armazenam vários elementos, mas não definem comportamento,
como as classes e os structs fazem. É fácil e sucinto usar tuplas para esses tipos. Você obtém todos os
benefícios da verificação de tipo estático, sem precisar criar tipos usando a sintaxe de class ou de struct
mais detalhada. Mesmo assim, elas são mais úteis para métodos utilitários que são private ou internal .
Cria tipos definidos pelo usuário, tipos class ou struct , quando seus métodos públicos retornam um valor
que tem vários elementos.
Desconstruindo tuplas e outros tipos
31/10/2019 • 16 minutes to read • Edit Online
Uma tupla fornece uma maneira leve de recuperar vários valores de uma chamada de método. Mas depois de
recuperar a tupla, você precisa lidar com seus elementos individuais. Fazer isso para cada elemento é incômodo,
conforme mostra o exemplo a seguir. O método QueryCityData retorna uma tupla de 3 e cada um de seus
elementos é atribuído a uma variável em uma operação separada.
using System;
Recuperar vários valores de propriedade e de campo de um objeto pode ser igualmente complicado: é necessário
atribuir um valor de campo ou de propriedade a uma variável, membro por membro.
Começando com o C# 7.0, você pode recuperar vários elementos de uma tupla ou recuperar vários valores
calculados, de campo e de propriedade de um objeto em uma única operação deconstruct. Quando você
desconstrói uma tupla, você atribui os elementos dela a variáveis individuais. Quando você desconstrói um objeto,
você atribui os elementos dela a variáveis individuais.
Você pode usar a palavra-chave var de modo que o C# infira o tipo de cada variável. Você coloca a
palavra-chave var fora dos parênteses. O exemplo a seguir usa a inferência de tipos ao desconstruir a
tupla de 3 retornada pelo método QueryCityData .
Você também pode usar a palavra-chave var individualmente com qualquer uma ou todas as declarações
de variável dentro dos parênteses.
Observe que você não poderá especificar um tipo específico fora dos parênteses, mesmo se todos os campos na
tupla tiverem o mesmo tipo. Isso gera o erro do compilador CS8136, "O formulário de desconstrução 'var (...)' não
permite um tipo específico para 'var'.".
Observe que você também deve atribuir cada elemento da tupla a uma variável. Se você omitir qualquer elemento,
o compilador gerará o erro CS8132, "Não é possível desconstruir uma tupla de 'x' elementos em 'y' variáveis."
Observe que não é possível combinar declarações e as atribuições com as variáveis existentes do lado esquerdo de
uma desconstrução. O compilador gera o erro CS8184, "uma desconstrução não pode combinar declarações e
expressões no lado esquerdo." quando os membros incluem variáveis recém-declaradas e existentes.
O exemplo a seguir ilustra o uso de tuplas com descartes. O método QueryCityDataForYears a seguir retorna uma
tupla de 6 com o nome de uma cidade, sua área, um ano, a população da cidade nesse ano, um segundo ano e
população da cidade nesse segundo ano. O exemplo mostra a alteração na população entre esses dois anos. Entre
os dados disponíveis da tupla, não estamos preocupados com a área da cidade e sabemos o nome da cidade e as
duas datas em tempo de design. Como resultado, estamos interessados apenas nos dois valores de população
armazenados na tupla e podemos lidar com seus valores restantes como descartes.
using System;
using System.Collections.Generic;
private static (string, double, int, int, int, int) QueryCityDataForYears(string name, int year1, int
year2)
{
int population1 = 0, population2 = 0;
double area = 0;
public void Deconstruct(out string fname, out string mname, out string lname)
Em seguida, você pode desconstruir uma instância da classe Person denominada p com uma atribuição
semelhante à seguinte:
O exemplo a seguir sobrecarrega o método Deconstruct para retornar várias combinações de propriedades de
um objeto Person . As sobrecargas individuais retornam:
Um nome e um sobrenome.
Um nome, um sobrenome e um segundo nome.
Um nome, um sobrenome, um nome de cidade e um nome de estado.
using System;
public void Deconstruct(out string fname, out string mname, out string lname)
{
fname = FirstName;
mname = MiddleName;
lname = LastName;
}
Já que você pode sobrecarregar o método Deconstruct para refletir os grupos de dados que geralmente são
extraídos de um objeto, você deve ter cuidado ao definir métodos Deconstruct com assinaturas que são diferentes
e não ambíguas. Vários métodos Deconstruct que têm o mesmo número de parâmetros out ou com o mesmo
número e tipo de parâmetros out em uma ordem diferente podem causar confusão.
O método Deconstruct sobrecarregado no exemplo a seguir ilustra uma possível fonte de confusão. A primeira
sobrecarga retorna o primeiro nome, o segundo nome, o sobrenome e idade de um objeto Person , nessa ordem.
A segunda sobrecarga retorna informações de nome apenas junto com a renda anual, mas o nome, o segundo
nome e o sobrenome estão em uma ordem diferente. Isso torna fácil confundir a ordem dos argumentos ao
desconstruir uma instância de Person .
using System;
public void Deconstruct(out string fname, out string mname, out string lname, out int age)
{
fname = FirstName;
mname = MiddleName;
lname = LastName;
age = DateTime.Now.Year - DateOfBirth.Year;
public void Deconstruct(out string lname, out string fname, out string mname, out decimal income)
{
fname = FirstName;
mname = MiddleName;
lname = LastName;
income = AnnualIncome;
}
}
using System;
using System.Collections.Generic;
using System.Reflection;
if (getter != null)
{
if (getter.IsPublic)
getAccessTemp = "public";
else if (getter.IsPrivate)
getAccessTemp = "private";
else if (getter.IsAssembly)
getAccessTemp = "internal";
else if (getter.IsFamily)
getAccessTemp = "protected";
else if (getter.IsFamilyOrAssembly)
getAccessTemp = "protected internal";
}
if (setter != null)
{
if (setter.IsPublic)
setAccessTemp = "public";
else if (setter.IsPrivate)
setAccessTemp = "private";
else if (setter.IsAssembly)
setAccessTemp = "internal";
else if (setter.IsFamily)
setAccessTemp = "protected";
else if (setter.IsFamilyOrAssembly)
setAccessTemp = "protected internal";
}
if (!hasGetAndSet | sameAccess)
{
Console.WriteLine(accessibility);
}
else
{
Console.WriteLine($"\n The get accessor: {getAccessibility}");
Console.WriteLine($" The set accessor: {setAccessibility}");
}
}
}
// The example displays the following output:
// The System.DateTime.Now property:
// PropertyType: DateTime
// Static: True
// Read-only: True
// Indexed: False
//
// Accessibility of the System.Collections.Generic.List`1.Item property: public
// Accessibility of the System.Collections.Generic.List`1.Item property: public
Consulte também
Descarta
Tuplas
Interfaces (Guia de Programação em C#)
31/10/2019 • 8 minutes to read • Edit Online
Uma interface contém definições para um grupo de funcionalidades relacionadas que uma classe não abstrata ou
uma struct deve implementar.
Usando interfaces, você pode, por exemplo, incluir o comportamento de várias fontes em uma classe. Essa
funcionalidade é importante em C# porque a linguagem não dá suporte a várias heranças de classes. Além disso,
use uma interface se você deseja simular a herança para structs, pois eles não podem herdar de outro struct ou
classe.
Você define uma interface usando a palavra-chave interface. como mostrado no exemplo a seguir.
interface IEquatable<T>
{
bool Equals(T obj);
}
O nome da struct deve ser um nome do identificador válido em C#. Por convenção, os nomes de interface
começam com uma letra maiúscula I .
Qualquer classe ou struct que implemente a interface IEquatable<T> deve conter uma definição para um método
Equals que corresponda à assinatura que a interface especifica. Como resultado, você pode contar com uma classe
que implementa IEquatable<T> para conter um método Equals com o qual uma instância da classe pode
determinar se é igual a outra instância da mesma classe.
A definição de IEquatable<T> não fornece uma implementação para Equals . Uma classe ou estrutura pode
implementar várias interfaces, mas uma classe só pode herdar de uma única classe.
Para obter mais informações sobre classes abstratas, consulte Classes e membros de classes abstratos e lacrados.
As interfaces podem conter métodos, propriedades, eventos, indexadores ou qualquer combinação desses quatro
tipos de membro. Para obter links para exemplos, consulte as seções relacionadas. Uma interface não pode conter
constantes, campos, operadores, construtores de instância, finalizadores ou tipos. Os membros da interface são
automaticamente públicos e eles não podem incluir nenhum modificador de acesso. Os membros também não
podem ser estáticos.
Para implementar um membro de interface, o membro correspondente da classe de implementação deve ser
público, não estático e ter o mesmo nome e assinatura do membro de interface.
Quando uma classe ou struct implementa uma interface, a classe ou o struct deve fornecer uma implementação
para todos os membros que a interface define. A interface não fornece nenhuma funcionalidade que uma classe ou
um struct possa herdar da forma que ela pode herdar a funcionalidade da classe base. No entanto, se uma classe
base implementa uma interface, qualquer classe que é derivada da classe base herda essa implementação.
O exemplo a seguir mostra uma implementação da interface IEquatable<T>. A classe de implementação, Car ,
deverá fornecer uma implementação do método Equals.
public class Car : IEquatable<Car>
{
public string Make {get; set;}
public string Model { get; set; }
public string Year { get; set; }
As propriedades e os indexadores de uma classe podem definir acessadores extras para uma propriedade ou o
indexador que é definido em uma interface. Por exemplo, uma interface pode declarar uma propriedade que tem
um acessador get. A classe que implementa a interface pode declarar a mesma propriedade tanto com um
acessador get quanto com um set. No entanto, se a propriedade ou o indexador usa a implementação explícita, os
acessadores devem corresponder. Para obter mais informações sobre a implementação explícita, consulte
Implementação de interface explícita e Propriedades da interface.
As interfaces podem herdar de outras interfaces. Uma classe pode incluir uma interface várias vezes por meio das
classes base que ela herda ou por meio de interfaces que outras interfaces herdam. No entanto, a classe poderá
fornecer uma implementação de uma interface apenas uma vez e somente se a classe declarar a interface como
parte da definição de classe ( class ClassName : InterfaceName ). Se a interface é herdada porque é herdada de uma
classe base que implementa a interface, a classe base fornece a implementação dos membros da interface. No
entanto, a classe derivada pode reimplementar qualquer membro de interface virtual em vez de usar a
implementação herdada.
Uma classe base também pode implementar membros de interface usando membros virtuais. Nesse caso, uma
classe derivada pode alterar o comportamento da interface substituindo os membros virtuais. Para obter mais
informações sobre membros virtuais, consulte Polimorfismo.
Resumo de interfaces
Uma interface tem as propriedades a seguir:
Uma interface é como uma classe base abstrata que contém apenas membros abstratos. Qualquer classe ou
struct que implementa a interface deve implementar todos os seus membros.
Uma interface não pode ser instanciada diretamente. Seus membros são implementados por qualquer classe ou
struct que implemente a interface.
As interfaces podem conter propriedades, indexadores, métodos e eventos.
As interfaces não têm implementações de métodos.
Uma classe ou struct pode implementar várias interfaces. Uma classe pode herdar uma classe base e também
implementar uma ou mais interfaces.
Nesta seção
Implementação de interface explícita
Explica como criar um membro da classe que é específico para uma interface.
Como implementar membros de interface explicitamente
Fornece um exemplo de como implementar explicitamente membros de interfaces.
Como implementar membros de duas interfaces explicitamente
Fornece um exemplo de como implementar explicitamente membros de interfaces com herança.
Seções relacionadas
Propriedades de interface
Indexadores em interfaces
Como implementar eventos de interface
Classes e Structs
Herança
Métodos
Polimorfismo
Classes e membros de classes abstract e sealed
Propriedades
Eventos
Indexadores
Consulte também
Guia de Programação em C#
Herança
Nomes de identificadores
Métodos
31/10/2019 • 37 minutes to read • Edit Online
Um método é um bloco de código que contém uma série de instruções. Um programa faz com que as instruções
sejam executadas chamando o método e especificando os argumentos de método necessários. No C#, todas as
instruções executadas são realizadas no contexto de um método. O método Main é o ponto de entrada para todos
os aplicativos C# e é chamado pelo CLR (Common Language Runtime) quando o programa é iniciado.
NOTE
Este tópico aborda os métodos nomeados. Para obter informações sobre funções anônimas, consulte Funções anônimas.
Assinaturas de método
Os métodos são declarados em uma class ou struct especificando:
Um nível de acesso opcional, como public ou private . O padrão é private .
Modificadores opcionais como abstract ou sealed .
O valor retornado ou void se o método não tiver nenhum.
O nome do método.
Quaisquer parâmetros de método. Os parâmetros de método estão entre parênteses e separados por vírgulas.
Parênteses vazios indicam que o método não requer parâmetros.
Essas partes juntas formam a assinatura do método.
NOTE
Um tipo de retorno de um método não faz parte da assinatura do método para fins de sobrecarga de método. No entanto,
ele faz parte da assinatura do método ao determinar a compatibilidade entre um delegado e o método para o qual ele
aponta.
O exemplo a seguir define uma classe chamada Motorcycle que contém cinco métodos:
using System;
Invocação de método
Os métodos podem ser de instância ou estáticos. Invocar um método de instância requer que você crie uma
instância de um objeto e chame o método nesse objeto. Um método de instância opera nessa instância e seus
dados. Você invoca um método estático referenciando o nome do tipo ao qual o método pertence; os métodos
estáticos não operam nos dados da instância. Tentar chamar um método estático por meio de uma instância do
objeto gera um erro do compilador.
Chamar um método é como acessar um campo. Após o nome do objeto (se você estiver chamando um método de
instância) ou o nome do tipo (se você estiver chamando um método static ), adicione um ponto, o nome do
método e parênteses. Os argumentos são listados dentro dos parênteses e são separados por vírgulas.
A definição do método especifica os nomes e tipos de quaisquer parâmetros obrigatórios. Quando um chamador
invoca o método, ele fornece valores concretos, chamados argumentos, para cada parâmetro. Os argumentos
devem ser compatíveis com o tipo de parâmetro, mas o nome do argumento, se for usado no código de chamada,
não precisa ser o mesmo que o parâmetro denominado definido no método. No exemplo a seguir, o método
Square inclui um único parâmetro do tipo int chamado i. A primeira chamada do método passa para o método
Square uma variável do tipo int chamada num, a segunda, uma constante numérica e a terceira, uma expressão.
A forma mais comum de invocação de método usa argumentos posicionais, ela fornece os argumentos na mesma
ordem que os parâmetros de método. Os métodos da classe Motorcycle , podem, portanto, ser chamados como
no exemplo a seguir. A chamada para o método Drive , por exemplo, inclui dois argumentos que correspondem
aos dois parâmetros na sintaxe do método. O primeiro se torna o valor do parâmetro miles , o segundo o valor
do parâmetro speed .
class TestMotorcycle : Motorcycle
{
public override double GetTopSpeed()
{
return 108.4;
}
moto.StartEngine();
moto.AddGas(15);
moto.Drive(5, 20);
double speed = moto.GetTopSpeed();
Console.WriteLine("My top speed is {0}", speed);
}
}
Você também pode usar argumentos nomeados em vez de argumentos posicionais ao invocar um método. Ao
usar argumentos nomeados, você especifica o nome do parâmetro seguido por dois pontos (":") e o argumento.
Os argumentos do método podem aparecer em qualquer ordem, desde que todos os argumentos necessários
estejam presentes. O exemplo a seguir usa argumentos nomeados para invocar o método TestMotorcycle.Drive .
Neste exemplo, os argumentos nomeados são passados na ordem oposta da lista de parâmetros do método.
using System;
Você pode invocar um método usando argumentos posicionais e argumentos nomeados. No entanto, um
argumento posicional não pode seguir um argumento nomeado. O exemplo a seguir invoca o método
TestMotorcycle.Drive do exemplo anterior usando um argumento posicional e um argumento nomeado.
using System;
Tipos podem substituir membros herdados usando a palavra-chave override e fornecendo uma implementação
para o método substituído. A assinatura do método precisa ser igual à do método substituído. O exemplo a seguir
é semelhante ao anterior, exceto que ele substitui o método Equals(Object). (Ele também substitui o método
GetHashCode(), uma vez que os dois métodos destinam-se a fornecer resultados consistentes.)
using System;
Passando parâmetros
Os tipos no C# são tipos de valor ou tipos de referência. Para obter uma lista de tipos de valor internos, consulte
Tipos e variáveis. Por padrão, os tipos de referência e tipos de valor são passados para um método por valor.
Passando parâmetros por valor
Quando um tipo de valor é passado para um método por valor, uma cópia do objeto, em vez do próprio objeto, é
passada para o método. Portanto, as alterações no objeto do método chamado não têm efeito no objeto original
quando o controle retorna ao chamador.
O exemplo a seguir passa um tipo de valor para um método por valor e o método chamado tenta alterar o valor
do tipo de valor. Ele define uma variável do tipo int , que é um tipo de valor, inicializa o valor para 20 e o passa
para um método chamado ModifyValue que altera o valor da variável para 30. No entanto, quando o método
retorna, o valor da variável permanece inalterado.
using System;
Quando um objeto do tipo de referência é passado para um método por valor, uma referência ao objeto é passada
por valor. Ou seja, o método recebe não o objeto em si, mas um argumento que indica o local do objeto. Se você
alterar um membro do objeto usando essa referência, a alteração será refletida no objeto quando o controle
retornar para o método de chamada. No entanto, substituir o objeto passado para o método não tem efeito no
objeto original quando o controle retorna para o chamador.
O exemplo a seguir define uma classe (que é um tipo de referência) chamada SampleRefType . Ele cria uma
instância de um objeto SampleRefType , atribui 44 ao seu campo value e passa o objeto para o método
ModifyObject . Este exemplo faz essencialmente a mesma coisa que o exemplo anterior, ele passa um argumento
por valor para um método. Mas como um tipo de referência é usado, o resultado é diferente. A modificação feita
em ModifyObject para o campo obj.value também muda o campo value do argumento, rt , no método Main
para 33, como a saída do exemplo mostra.
using System;
using System;
Um padrão comum que usa parâmetros pela referência envolve a troca os valores das variáveis. Você passa duas
variáveis para um método por referência e o método troca seus conteúdos. O exemplo a seguir troca valores
inteiros.
using System;
using System;
using System.Linq;
class Example
{
static void Main()
{
string fromArray = GetVowels(new[] { "apple", "banana", "pear" });
Console.WriteLine($"Vowels from array: '{fromArray}'");
Se um método inclui parâmetros obrigatórios e opcionais, os parâmetros opcionais são definidos no final da lista
de parâmetros, após todos os parâmetros obrigatórios.
O exemplo a seguir define um método, ExampleMethod , que tem um parâmetro obrigatório e dois opcionais.
using System;
Se um método com vários argumentos opcionais for invocado usando argumentos posicionais, o chamador
deverá fornecer um argumento para todos os parâmetros opcionais do primeiro ao último para o qual um
argumento é fornecido. No caso do método ExampleMethod , por exemplo, se o chamador fornecer um argumento
para o parâmetro description , ele deverá fornecer também um para o parâmetro optionalInt .
opt.ExampleMethod(2, 2, "Addition of 2 and 2"); é uma chamada de método válida,
opt.ExampleMethod(2, , "Addition of 2 and 0"); gera um erro do compilador de “Argumento ausente”.
Se um método for chamado usando argumentos nomeados ou uma combinação de argumentos posicionais e
nomeados, o chamador poderá omitir todos os argumentos após o último argumento posicional na chamada do
método.
A exemplo a seguir chama o método ExampleMethod três vezes. As duas primeiras chamadas de método usam
argumentos posicionais. O primeiro omite ambos os argumentos opcionais, enquanto o segundo omite o último
argumento. A terceira chamada de método fornece um argumento posicional para o parâmetro obrigatório, mas
usa um argumento nomeado para fornecer um valor para o parâmetro description enquanto omite o argumento
optionalInt .
O uso de parâmetros opcionais afeta a resolução de sobrecarga ou a maneira em que o compilador C# determina
qual sobrecarga específica deve ser invocada pela invocada de método, da seguinte maneira:
Um método, indexador ou construtor é um candidato para a execução se cada um dos parâmetros é opcional
ou corresponde, por nome ou posição, a um único argumento na instrução de chamada e esse argumento pode
ser convertido para o tipo do parâmetro.
Se mais de um candidato for encontrado, as regras de resolução de sobrecarga de conversões preferenciais
serão aplicadas aos argumentos que são especificados explicitamente. Os argumentos omitidos para
parâmetros opcionais são ignorados.
Se dois candidatos são considerados igualmente bons, a preferência vai para um candidato que não tem
parâmetros opcionais para os quais argumentos foram omitidos na chamada. Esta é uma consequência da
preferência geral na resolução de sobrecarga de candidatos que têm menos parâmetros.
Valores de retorno
Os métodos podem retornar um valor para o chamador. Se o tipo de retorno (o tipo listado antes do nome do
método) não for void , o método poderá retornar o valor usando palavra-chave return . Uma instrução com a
palavra-chave return seguida por uma variável, constante ou expressão que corresponde ao tipo de retorno
retornará esse valor para o chamador do método. Métodos com um tipo de retorno não nulo devem usar a
palavra-chave return para retornar um valor. A palavra-chave return também interrompe a execução do
método.
Se o tipo de retorno for void , uma instrução return sem um valor ainda será útil para interromper a execução
do método. Sem a palavra-chave return , a execução do método será interrompida quando chegar ao final do
bloco de código.
Por exemplo, esses dois métodos usam a palavra-chave return para retornar inteiros:
class SimpleMath
{
public int AddTwoNumbers(int number1, int number2)
{
return number1 + number2;
}
Para usar um valor retornado de um método, o método de chamada pode usar a chamada de método em si em
qualquer lugar que um valor do mesmo tipo seria suficiente. Você também pode atribuir o valor retornado a uma
variável. Por exemplo, os dois exemplos de código a seguir obtêm a mesma meta:
Usar uma variável local, nesse caso, result , para armazenar um valor é opcional. Isso pode ajudar a legibilidade
do código ou pode ser necessário se você precisar armazenar o valor original do argumento para todo o escopo
do método.
Às vezes, você deseja que seu método retorne mais de um único valor. A partir do C# 7.0, você pode fazer isso
facilmente usando tipos de tupla e literais de tupla. O tipo de tupla define os tipos de dados dos elementos da
tupla. Os literais de tupla fornecem os valores reais da tupla retornada. No exemplo a seguir,
(string, string, string, int) define o tipo de tupla que é retornado pelo método GetPersonalInfo . A expressão
(per.FirstName, per.MiddleName, per.LastName, per.Age) é a tupla literal, o método retorna o nome, o nome do
meio e o sobrenome, juntamente com a idade, de um objeto PersonInfo .
O chamador pode então consumir a tupla retornada com o código semelhante ao seguinte:
Os nomes também podem ser atribuídos aos elementos da tupla na definição de tipo de tupla. O exemplo a seguir
mostra uma versão alternativa do método GetPersonalInfo que usa elementos nomeados:
public (string FName, string MName, string LName, int Age) GetPersonalInfo(string id)
{
PersonInfo per = PersonInfo.RetrieveInfoById(id);
return (per.FirstName, per.MiddleName, per.LastName, per.Age);
}
A chamada anterior para o método GetPersonInfo pode ser modificada da seguinte maneira:
Se um método passa uma matriz como um argumento e modifica o valor de elementos individuais, o método não
precisa retornar a matriz, embora você possa optar por fazer isso para obter um bom estilo ou um fluxo de valores
funcional. Isso ocorre porque o C# passa todos os tipos de referência por valor e o valor de uma referência de
matriz é o ponteiro para a matriz. No exemplo a seguir, as alterações no conteúdo da matriz values realizados
pelo método DoubleValues são observáveis por qualquer código que faz referência à matriz.
using System;
Métodos de extensão
Normalmente, há duas maneiras de adicionar um método a um tipo existente:
Modificar o código-fonte para esse tipo. Você não pode fazer isso, é claro, se não possui o código-fonte do tipo.
E isso se torna uma alteração significativa se você também adicionar campos de dados privados para dar
suporte ao método.
Definir o novo método em uma classe derivada. Não é possível adicionar um método dessa forma usando a
herança para outros tipos, como estruturas e enumerações. Isso também não pode ser usado para “adicionar”
um método a uma classe selada.
Os métodos de extensão permitem que você “adicione” um método a um tipo existente sem modificar o tipo em si
ou implementar o novo método em um tipo herdado. O método de extensão também não precisa residir no
mesmo assembly que o tipo que ele estende. Você chama um método de extensão como se fosse um membro
definido de um tipo.
Para obter mais informações, consulte Métodos de extensão.
Métodos assíncronos
Usando o recurso async, você pode invocar métodos assíncronos sem usar retornos de chamada explícitos ou
dividir manualmente seu código entre vários métodos ou expressões lambda.
Se marcar um método com o modificador async, você poderá usar o operador await no método. Quando o
controle atingir uma expressão await no método assíncrono, o controle retornará para o chamador se a tarefa
aguardada não estiver concluída e o progresso no método com a palavra-chave await será suspenso até a tarefa
aguardada ser concluída. Quando a tarefa for concluída, a execução poderá ser retomada no método.
NOTE
Um método assíncrono retorna para o chamador quando encontra o primeiro objeto esperado que ainda não está completo
ou chega ao final do método assíncrono, o que ocorrer primeiro.
Um método assíncrono pode ter um tipo de retorno Task<TResult>, Task ou void . O tipo de retorno void é
usado principalmente para definir manipuladores de eventos, nos quais o tipo de retorno void é necessário. Um
método assíncrono que retorna void não pode ser aguardado e o chamador de um método de retorno nulo não
pode capturar as exceções que esse método gera. Começando com o C# 7.0, um método assíncrono pode ter
qualquer tipo de retorno como os de tarefa.
No exemplo a seguir, DelayAsync é um método assíncrono que contém uma instrução return que retorna um
inteiro. Como é um método assíncrono, sua declaração de método deve ter um tipo de retorno de Task<int> .
Como o tipo de retorno é Task<int> , a avaliação da expressão await em DoSomethingAsync produz um inteiro,
como a instrução int result = await delayTask a seguir demonstra.
using System;
using System.Diagnostics;
using System.Threading.Tasks;
// Output:
// Result: 5
}
// The example displays the following output:
// Result: 5
Um método assíncrono não pode declarar os parâmetros in, ref nem out, mas pode chamar métodos que tenham
esses parâmetros.
Para obter mais informações sobre os métodos assíncronos, consulte Programação assíncrona com async e await,
Fluxo de controle em programas assíncronos e Tipos de retorno assíncronos.
public Point Move(int dx, int dy) => new Point(x + dx, y + dy);
public void Print() => Console.WriteLine(First + " " + Last);
// Works with operators, properties, and indexers too.
public static Complex operator +(Complex a, Complex b) => a.Add(b);
public string Name => First + " " + Last;
public Customer this[long id] => store.LookupCustomer(id);
Se o método retornar void ou for um método assíncrono, o corpo do método deverá ser uma expressão de
instrução (igual aos lambdas). Para propriedades e indexadores, eles devem ser somente leitura e não usar a
palavra-chave do acessador get .
Iterators
Um iterador realiza uma iteração personalizada em uma coleção, como uma lista ou uma matriz. Um iterador usa
a instrução yield return para retornar um elemento de cada vez. Quando uma instrução yield return for atingida,
o local atual será lembrado para que o chamador possa solicitar o próximo elemento na sequência.
O tipo de retorno de um iterador pode ser IEnumerable, IEnumerable<T>, IEnumerator ou IEnumerator<T>.
Para obter mais informações, consulte Iteradores.
Consulte também
Modificadores de acesso
Classes static e membros de classes static
Herança
Classes e membros de classes abstract e sealed
params
out
ref
in
Passando parâmetros
Expressões lambda (Guia de Programação em C#)
04/11/2019 • 19 minutes to read • Edit Online
Uma expressão lambda é uma expressão de uma das duas seguintes formas:
Expressão lambda que tem uma expressão como corpo:
Use o operador de declaração lambda => para separar a lista de parâmetros de lambda do corpo. Para criar uma
expressão lambda, especifique os parâmetros de entrada (se houver) no lado esquerdo do operador lambda e uma
expressão ou um bloco de instrução do outro lado.
Qualquer expressão lambda pode ser convertida para um tipo delegado. O tipo delegado no qual uma expressão
lambda pode ser convertida é definido pelos tipos de parâmetros e pelo valor retornado. Se uma expressão lambda
não retornar um valor, ela poderá ser convertida em um dos tipos delegados Action ; caso contrário, ela poderá ser
convertida em um dos tipos delegados Func . Por exemplo, uma expressão lambda que tem dois parâmetros e não
retorna nenhum valor pode ser convertida em um delegado Action<T1,T2>. Uma expressão lambda que tem um
parâmetro e retorna um valor pode ser convertida em um delegado Func<T,TResult>. No seguinte exemplo, a
expressão lambda x => x * x , que especifica um parâmetro chamado x e retorna o valor de x quadrado, é
atribuída a uma variável de um tipo delegado:
As expressões lambdas também podem ser convertidas nos tipos de árvore de expressão, como mostra o seguinte
exemplo:
Use expressões lambda em qualquer código que exija instâncias de tipos delegados ou árvores de expressão, por
exemplo, como um argumento ao método Task.Run(Action) para passar o código que deve ser executado em
segundo plano. Você também pode usar expressões lambda ao escrever LINQ no C# , como mostra o exemplo a
seguir:
int[] numbers = { 2, 3, 4, 5 };
var squaredNumbers = numbers.Select(x => x * x);
Console.WriteLine(string.Join(" ", squaredNumbers));
// Output:
// 4 9 16 25
Quando você usa a sintaxe baseada em método para chamar o método Enumerable.Select na classe
System.Linq.Enumerable, por exemplo, no LINQ to Objects e no LINQ to XML, o parâmetro é um tipo delegado
System.Func<T,TResult>. Quando você chama o método Queryable.Select na classe System.Linq.Queryable, por
exemplo, no LINQ to SQL, o tipo de parâmetro é um tipo de árvore de expressão
Expression<Func<TSource,TResult>> . Em ambos os casos, você pode usar a mesma expressão lambda para
especificar o valor do parâmetro. Isso faz com que as duas chamadas Select pareçam semelhantes, embora, na
verdade, o tipo de objetos criado dos lambdas seja diferente.
Lambdas de expressão
Uma expressão lambda com uma expressão no lado direito do operador => é chamada de lambda de expressão.
Os lambdas de expressão são usados amplamente na construção de árvores de expressão. Uma expressão lambda
retorna o resultado da expressão e tem o seguinte formato básico:
Os parênteses serão opcionais somente se o lambda tiver um parâmetro de entrada; caso contrário, eles serão
obrigatórios.
Especifique parâmetros de entrada zero com parênteses vazios:
Dois ou mais parâmetros de entrada são separados por vírgulas e envolvidos por parênteses:
Às vezes, é difícil ou impossível para o compilador inferir os tipos de entrada. Você pode especificar os tipos de
maneira explícita conforme mostrado neste exemplo:
Os tipos de parâmetro de entrada devem ser todos explícitos ou implícitos; caso contrário, ocorrerá o erro CS0748
de compilador.
O corpo de um lambda de expressão pode consistir em uma chamada de método. No entanto, se você estiver
criando árvores de expressão que serão avaliadas fora contexto do .NET Common Language Runtime, como no
SQL Server, você não deverá usar chamadas de método em lambdas de expressão. Os métodos não terão
significado fora do contexto do .NET Common Language Runtime.
Lambdas de instrução
Um lambda de instrução lembra um lambda de expressão, exceto que as instruções estão incluídas entre chaves:
O corpo de uma instrução lambda pode consistir de qualquer número de instruções; no entanto, na prática,
normalmente não há mais de duas ou três.
Action<string> greet = name =>
{
string greeting = $"Hello {name}!";
Console.WriteLine(greeting);
};
greet("World");
// Output:
// Hello World!
Os lambdas de instrução não podem ser usados para criar árvores de expressão.
Lambdas assíncronos
Você pode facilmente criar expressões e instruções lambda que incorporem processamento assíncrono, ao usar as
palavras-chaves async e await. Por exemplo, o exemplo do Windows Forms a seguir contém um manipulador de
eventos que chama e espera um método assíncrono ExampleMethodAsync .
Você pode adicionar o mesmo manipulador de eventos ao usar um lambda assíncrono. Para adicionar esse
manipulador, adicione um modificador async antes da lista de parâmetros lambda, como mostra o exemplo a
seguir:
Func<(int, int, int), (int, int, int)> doubleThem = ns => (2 * ns.Item1, 2 * ns.Item2, 2 * ns.Item3);
var numbers = (2, 3, 4);
var doubledNumbers = doubleThem(numbers);
Console.WriteLine($"The set {numbers} doubled: {doubledNumbers}");
// Output:
// The set (2, 3, 4) doubled: (4, 6, 8)
Normalmente, os campos de uma tupla são nomeados Item1 , Item2 , etc. No entanto, você pode definir uma tupla
com componentes nomeados, como faz o exemplo a seguir.
Func<(int n1, int n2, int n3), (int, int, int)> doubleThem = ns => (2 * ns.n1, 2 * ns.n2, 2 * ns.n3);
var numbers = (2, 3, 4);
var doubledNumbers = doubleThem(numbers);
Console.WriteLine($"The set {numbers} doubled: {doubledNumbers}");
Para saber mais sobre tuplas C#, confira o artigo sobre tipos de tuplas C#.
O delegado pode ser instanciado como um Func<int, bool> , em que int é um parâmetro de entrada e bool é o
valor de retorno. O valor de retorno é sempre especificado no último parâmetro de tipo. Por exemplo,
Func<int, string, bool> define um delegado com dois parâmetros de entrada, int e string , e um tipo de
retorno de bool . O delegado Func a seguir, quando é invocado, retornará um valor booliano que indica se o
parâmetro de entrada é ou não igual a cinco:
Você também pode fornecer uma expressão lambda quando o tipo de argumento é um Expression<TDelegate>.
Por exemplo, nos operadores de consulta padrão que são definidos no tipo Queryable. Quando você especifica um
argumento Expression<TDelegate>, o lambda é compilado em uma árvore de expressão.
O exemplo a seguir usa o operador padrão de consulta Count:
int[] numbers = { 5, 4, 1, 3, 9, 8, 6, 7, 2, 0 };
int oddNumbers = numbers.Count(n => n % 2 == 1);
Console.WriteLine($"There are {oddNumbers} odd numbers in {string.Join(" ", numbers)}");
O compilador pode inferir o tipo de parâmetro de entrada ou você também pode especificá-lo explicitamente. Essa
expressão lambda em particular conta esses inteiros ( n ) que, quando dividida por dois, tem um resto 1.
O exemplo a seguir gera uma sequência que contém todos os elementos da matriz numbers que precedem o 9,
porque esse é o primeiro número na sequência que não satisfaz a condição:
int[] numbers = { 5, 4, 1, 3, 9, 8, 6, 7, 2, 0 };
var firstNumbersLessThanSix = numbers.TakeWhile(n => n < 6);
Console.WriteLine(string.Join(" ", firstNumbersLessThanSix));
// Output:
// 5 4 1 3
O exemplo a seguir especifica vários parâmetros de entrada, colocando-os entre parênteses. O método retorna
todos os elementos na matriz numbers até encontrar um número cujo valor seja inferior à sua posição ordinal na
matriz:
int[] numbers = { 5, 4, 1, 3, 9, 8, 6, 7, 2, 0 };
var firstSmallNumbers = numbers.TakeWhile((n, index) => n >= index);
Console.WriteLine(string.Join(" ", firstSmallNumbers));
// Output:
// 5 4
updateCapturedLocalVariable = x =>
{
j = x;
bool result = j > input;
Console.WriteLine($"{j} is greater than {input}: {result}");
};
isEqualToCapturedLocalVariable = x => x == j;
int gameInput = 5;
game.Run(gameInput);
int anotherJ = 3;
game.updateCapturedLocalVariable(anotherJ);
Especificação da linguagem C#
Para obter mais informações, confira a seção Expressões de função anônima da Especificação da linguagem C#.
Consulte também
Guia de Programação em C#
LINQ (Consulta Integrada à Linguagem)
Árvores de Expressão
Funções locais comparadas com expressões lambda
Expressões lambda tipadas implicitamente
Exemplos de C# do Visual Studio 2008 (veja os arquivos de exemplo de consultas LINQ e programa XQuery)
Expressões lambda recursivas
Propriedades
30/10/2019 • 16 minutes to read • Edit Online
As propriedades são cidadãos de primeira classe no C#. A linguagem define uma sintaxe que permite aos
desenvolvedores escrever código que expresse sua intenção de design com precisão.
As propriedades se comportam como campos quando são acessadas. No entanto, diferentemente dos campos,
as propriedades são implementadas com acessadores, que definem as instruções que são executadas quando
uma propriedade é acessada ou atribuída.
Sintaxe de propriedade
A sintaxe para propriedades é uma extensão natural para os campos. Um campo define um local de
armazenamento:
Uma definição de propriedade contém declarações para um acessador get e set que recupera e atribui o valor
dessa propriedade:
A sintaxe mostrada acima é a sintaxe da propriedade automática. O compilador gera o local de armazenamento
para o campo que dá suporte à propriedade. O compilador também implementa o corpo dos acessadores get e
set .
Às vezes, você precisa inicializar uma propriedade para um valor diferente do padrão para seu tipo. O C# permite
isso definindo um valor após a chave de fechamento da propriedade. Você pode preferir que o valor inicial para a
propriedade FirstName seja a cadeia de caracteres vazia em vez de null . Você deve especificar isso conforme
mostrado abaixo:
A inicialização específica é mais útil para propriedades somente leitura, como você verá adiante neste artigo.
Você mesmo também pode definir o armazenamento, conforme mostrado abaixo:
public class Person
{
public string FirstName
{
get { return firstName; }
set { firstName = value; }
}
private string firstName;
// remaining implementation removed from listing
}
Quando uma implementação de propriedade é uma única expressão, você pode usar membros aptos para
expressão para o getter ou setter:
Essa sintaxe simplificada será usada quando aplicável ao longo deste artigo.
A definição da propriedade mostrada acima é uma propriedade de leitura/gravação. Observe a palavra-chave
value no acessador set. O acessador set sempre tem um parâmetro único chamado value . O acessador get
deve retornar um valor que seja conversível para o tipo da propriedade ( string , neste exemplo).
Essas são as noções básicas sobre a sintaxe. Há muitas variações diferentes que oferecem suporte a uma
variedade de linguagens de design diferentes. Vamos explorá-las e conhecer as opções de sintaxe para cada uma.
Cenários
Os exemplos acima mostraram um dos casos mais simples de definição de propriedade: uma propriedade de
leitura/gravação sem validação. Ao escrever o código que você deseja nos acessadores get e set , você pode
criar vários cenários diferentes.
Validação
Você pode escrever código no acessador set para garantir que os valores representados por uma propriedade
sejam sempre válidos. Por exemplo, suponha que uma regra para a classe Person é que o nome não pode ser
um espaço em branco. Você escreveria isso da seguinte maneira:
public class Person
{
public string FirstName
{
get => firstName;
set
{
if (string.IsNullOrWhiteSpace(value))
throw new ArgumentException("First name must not be blank");
firstName = value;
}
}
private string firstName;
// remaining implementation removed from listing
}
O exemplo anterior pode ser simplificado usando uma expressão throw como parte da validação de setter de
propriedade:
O exemplo acima aplica a regra de que o nome não pode ser em branco ou espaço em branco. Se um
desenvolvedor escreve
hero.FirstName = "";
Essa atribuição lança uma ArgumentException . Como um acessador set de propriedade deve ter um tipo de
retorno void, você relata erros no acessador set lançando uma exceção.
Você pode estender essa mesma sintaxe para qualquer coisa necessária em seu cenário. Você pode verificar as
relações entre diferentes propriedades ou validar em relação a qualquer condição externa. Todas as instruções de
C# válidas são válidas em um acessador de propriedade.
Somente leitura
Até aqui, todas as definições de propriedade que você viu são de propriedades de leitura/gravação com
acessadores públicos. Essa não é a única acessibilidade válida para as propriedades. Você pode criar
propriedades somente leitura ou dar acessibilidade diferente aos acessadores get e set. Suponha que sua classe
Person só deva habilitar a alteração do valor da propriedade FirstName em outros métodos naquela classe.
Você pode dar acessibilidade private ao acessador set, em vez de public :
Esse recurso é mais comumente usado para inicializar coleções que são expostas como propriedades somente
leitura:
Propriedades computadas
Uma propriedade não precisa simplesmente retornar o valor de um campo de membro. Você pode criar
propriedades que retornam um valor computado. Vamos expandir o objeto Person para retornar o nome
completo, computado pela concatenação dos nomes e sobrenomes:
O exemplo acima usa o recurso de interpolação de cadeia de caracteres para criar a cadeia de caracteres
formatada do nome completo.
Use também um membro com corpo da expressão, que fornece uma maneira mais sucinta de criar a propriedade
FullName computada:
public class Person
{
public string FirstName { get; set; }
Os membros com corpo da expressão usam a sintaxe expressão lambda para definir métodos que contêm uma
única expressão. Aqui, essa expressão retorna o nome completo do objeto person.
Propriedades avaliadas armazenadas em cache
Combine o conceito de uma propriedade computada com o armazenamento e crie uma propriedade avaliada
armazenada em cache. Por exemplo, você poderia atualizar a propriedade FullName para que a formatação da
cadeia de caracteres só acontecesse na primeira vez que ela foi acessada:
No entanto, o código acima contém um bug. Se o código atualizar o valor das propriedades FirstName ou
LastName , o campo fullName , anteriormente avaliado, será inválido. Modifique os acessadores set das
propriedades FirstName e LastName para que o campo fullName seja calculado novamente:
public class Person
{
private string firstName;
public string FirstName
{
get => firstName;
set
{
firstName = value;
fullName = null;
}
}
Esta versão final avalia a propriedade FullName apenas quando necessário. Se a versão calculada anteriormente
for válida, ela será usada. Se outra alteração de estado invalidar a versão calculada anteriormente, ela será
recalculada. Os desenvolvedores que usam essa classe não precisam saber dos detalhes da implementação.
Nenhuma dessas alterações internas afetam o uso do objeto Person. Esse é o motivo principal para o uso de
propriedades para expor os membros de dados de um objeto.
Anexando atributos a propriedades autoimplementadas
Do C# 7.3 em diante, atributos de campo podem ser anexados ao campo de suporte gerado pelo compilador em
propriedades autoimplementadas. Por exemplo, considere uma revisão da classe Person que adiciona uma
propriedade Id de inteiro exclusivo. Você escreve a propriedade Id usando uma propriedade
autoimplementada, mas o design não exige a persistência da propriedade Id . O NonSerializedAttribute pode
ser anexado apenas a campos, não a propriedades. Anexe o NonSerializedAttribute ao campo de suporte da
propriedade Id usando o especificador field: no atributo, conforme mostrado no seguinte exemplo:
[field:NonSerialized]
public int Id { get; set; }
O operador ?. é chamado de operador condicional nulo. Ele verifica uma referência nula antes de avaliar o lado
direito do operador. O resultado final é que, se não houver nenhum assinante para o evento PropertyChanged , o
código para acionar o evento não é executado. Ela lançaria uma NullReferenceException sem essa verificação,
nesse caso. Para obter mais informações, consulte events . Este exemplo também usa o novo operador nameof
para converter o símbolo de nome da propriedade em sua representação de texto. O uso de nameof pode
reduzir erros no local em que você digitou errado o nome da propriedade.
Novamente, a implementação de INotifyPropertyChanged é um exemplo de um caso em que você pode escrever
o código nos acessadores para dar suporte aos cenários necessários.
Resumindo
As propriedades são uma forma de campos inteligentes em uma classe ou objeto. De fora do objeto, elas
parecem como campos no objeto. No entanto, as propriedades podem ser implementadas usando a paleta
completa de funcionalidades do C#. Você pode fornecer validação, acessibilidade diferente, avaliação lenta ou
quaisquer requisitos necessários aos seus cenários.
Indexadores
30/10/2019 • 15 minutes to read • Edit Online
Os indexadores são semelhantes às propriedades. De muitas maneiras, os indexadores baseiam-se nos mesmos
recursos de linguagem que as propriedades. Os indexadores habilitam as propriedades indexadas: propriedades
referenciadas com o uso de um ou mais argumentos. Esses argumentos fornecem um índice em um conjunto de
valores.
Sintaxe do indexador
Você pode acessar um indexador por meio de um nome de variável e colchetes. Coloque os argumentos do
indexador dentro de colchetes:
Você declara os indexadores usando a palavra-chave this como o nome da propriedade e declarando os
argumentos entre colchetes. Essa declaração corresponderia à utilização mostrada no parágrafo anterior:
Neste exemplo inicial, você pode ver a relação entre a sintaxe das propriedades e dos indexadores. Essa analogia
impulsiona a maioria das regras de sintaxe para indexadores. Os indexadores podem ter qualquer modificador de
acesso válido (público, protegido interno, protegido, interno, particular ou protegido de forma privada). Eles podem
ser sealed, virtual ou abstract. Assim como acontece com as propriedades, você pode especificar modificadores de
acesso diferentes para os acessadores get e set em um indexador. Você também pode especificar indexadores
somente leitura (omitindo o acessador set) ou indexadores somente gravação (omitindo o acessador get).
Você pode aplicar aos indexadores quase tudo o que aprendeu ao trabalhar com propriedades. A única exceção a
essa regra são as propriedades autoimplementadas. O compilador não pode gerar sempre o armazenamento
correto para um indexador.
A presença dos argumentos para referenciar um item em um conjunto de itens distingue os indexadores das
propriedades. Você pode definir vários indexadores em um tipo, contanto que as listas de argumentos para cada
indexador seja exclusiva. Vamos explorar diferentes cenários em que você pode usar um ou mais indexadores em
uma definição de classe.
Cenários
Você deve definir indexadores em seu tipo quando a API do tipo modela alguma coleção na qual você define os
argumentos para essa coleção. Seu indexadores podem ou não mapear diretamente para os tipos de coleção que
fazem parte da estrutura principal do .NET. O tipo pode ter outras responsabilidades, além da modelagem de uma
coleção. Os indexadores permitem que você forneça a API que corresponda à abstração do tipo, sem expor os
detalhes internos de como os valores dessa abstração são armazenados ou computados.
Vamos examinar alguns dos cenários comuns de uso de indexadores. Você pode acessar a pasta de exemplo para
indexadores. Para obter instruções de download, consulte Exemplos e tutoriais.
Matrizes e vetores
Um dos cenários mais comuns para a criação de indexadores é quando seu tipo modela uma matriz ou um vetor.
Você pode criar um indexador para modelar uma lista ordenada de dados.
A vantagem de criar seu próprio indexador é que você pode definir o armazenamento dessa coleção para atender
às suas necessidades. Imagine um cenário em que seu tipo modela dados históricos que são muito grandes para
serem carregados na memória ao mesmo tempo. Você precisa carregar e descarregar seções da coleção com base
na utilização. O exemplo a seguir modela esse comportamento. Ele relata quantos pontos de dados existem. Ele
cria páginas para manter as seções de dados sob demanda. Ele remove páginas da memória a fim de liberar
espaço para as páginas necessárias para as solicitações mais recentes.
page[index] = value;
}
}
Você pode seguir esta linguagem de design para modelar qualquer tipo de coleção na qual há bons motivos para
não carregar todo o conjunto de dados em uma coleção na memória. Observe que a classe Page é uma classe
particular aninhada que não faz parte da interface pública. Esses detalhes estão ocultos de qualquer usuário dessa
classe.
Dicionários
Outro cenário comum é quando você precisa modelar um dicionário ou um mapa. Esse cenário é quando o seu
tipo armazena valores com base na chave, normalmente chaves de texto. Este exemplo cria um dicionário que
mapeia os argumentos de linha de comando para expressões lambda que gerenciam essas opções. O exemplo a
seguir mostra duas classes: uma classe ArgsActions que mapeia uma opção de linha de comando para um
delegado Action e uma ArgsProcessor , que usa a ArgsActions para executar cada Action , quando encontrar
essa opção.
}
public class ArgsActions
{
readonly private Dictionary<string, Action> argsActions = new Dictionary<string, Action>();
Neste exemplo, a coleção ArgsAction mapeia próximo à coleção subjacente. O get determina se uma opção
específica foi configurada. Se sim, ele retorna a Action associada a essa opção. Se não, ele retorna uma Action
que não faz nada. O acessador público não inclui um acessador set . Em vez disso, o design usa um método
público para a configuração de opções.
Mapas multidimensionais
Você pode criar indexadores que usam vários argumentos. Além disso, esses argumentos não estão restritos a
serem do mesmo tipo. Vamos analisar dois exemplos.
O primeiro exemplo mostra uma classe que gera valores para um conjunto de Mandelbrot. Para obter mais
informações sobre a matemática por trás desse conjunto, leia este artigo. O indexador usa dois duplos para definir
um ponto no plano X, Y. O acessador get calcula o número de iterações até que um ponto seja considerado como
fora do conjunto. Se o número máximo de iterações for atingido, o ponto está no conjunto e o valor da classe
maxIterations será retornado. (As imagens geradas por computador, popularizadas para o conjunto de Mandelbrot,
definem cores para o número de iterações necessárias para determinar que um ponto está fora do conjunto.
O conjunto de Mandelbrot define valores em cada coordenada (x,y) para valores de número real. Isso define um
dicionário que poderia conter um número infinito de valores. Portanto, não há armazenamento por trás desse
conjunto. Em vez disso, essa classe calcula o valor de cada ponto quando o código chama o acessador get . Não há
nenhum armazenamento subjacente usado.
Vamos examinar um último uso de indexadores, em que o indexador recebe vários argumentos de tipos diferentes.
Considere um programa que gerencia os dados históricos de temperatura. Esse indexador utiliza uma cidade e
uma data para definir ou obter as temperaturas máximas e mínimas desse local:
using DateMeasurements =
System.Collections.Generic.Dictionary<System.DateTime, IndexersSamples.Common.Measurements>;
using CityDataMeasurements =
System.Collections.Generic.Dictionary<string, System.Collections.Generic.Dictionary<System.DateTime,
IndexersSamples.Common.Measurements>>;
Este exemplo cria um indexador que mapeia dados meteorológicos de dois argumentos diferentes: uma cidade
(representada por uma string ) e uma data (representada por uma DateTime ). O armazenamento interno usa
duas classes Dictionary para representar o dicionário bidimensional. A API pública não representa mais o
armazenamento subjacente. Em vez disso, os recursos de linguagem dos indexadores permite que você crie uma
interface pública que representa a sua abstração, mesmo que o armazenamento subjacente deva usar diferentes
tipos principais de coleção.
Há duas partes desse código que podem não ser familiares para alguns desenvolvedores. Essas duas instruções
using :
criam um alias para um tipo genérico construído. Essas instruções habilitam o código a usar, mais adiante, os
nomes DateMeasurements e CityDateMeasurements mais descritivos, em vez da construção genérica de
Dictionary<DateTime, Measurements> e Dictionary<string, Dictionary<DateTime, Measurements> > . Esse constructo
exige o uso de nomes de tipo totalmente qualificados no lado direito do sinal = .
A segunda técnica é para remover as partes de hora de qualquer objeto DateTime usado para indexar na coleção.
A estrutura do .NET não inclui um tipo de Somente Data. Os desenvolvedores usam o tipo DateTime , mas usam a
propriedade Date para garantir que qualquer objeto DateTime daquele dia sejam iguais.
Resumindo
Você deve criar indexadores sempre que tiver um elemento semelhante a uma propriedade em sua classe, em que
essa propriedade representa não um único valor, mas uma coleção de valores em que cada item individual é
identificado por um conjunto de argumentos. Esses argumentos podem identificar exclusivamente qual item da
coleção deve ser referenciado. Os indexadores estendem o conceito de Propriedades, em que um membro é
tratado como um item de dados de fora da classe, mas como um método no interior. Os indexadores permitem que
os argumentos localizem um único item em uma propriedade que representa um conjunto de itens.
Descartes – Guia do C#
31/10/2019 • 12 minutes to read • Edit Online
Começando com o C# 7.0, o C# é compatível com descartes, que são variáveis temporárias e fictícias, não
utilizadas intencionalmente no código do aplicativo. Descartes são equivalentes a variáveis não atribuídas; eles
não têm um valor. Já que há apenas uma variável de descarte e essa variável pode nem mesmo ser de
armazenamento alocado, descartes podem reduzir as alocações de memória. Por deixarem clara a intenção do seu
código, eles melhoram a sua legibilidade e a facilidade de manutenção.
Você indica que uma variável é um descarte atribuindo a ela o sublinhado ( _ ) como seu nome. Por exemplo, a
chamada de método a seguir retorna uma tupla de 3 na qual o primeiro e o segundo valores são descartes, e area
é uma variável declarada anteriormente para ser definida para o terceiro componente correspondente retornado
por GetCityInformation:
private static (string, double, int, int, int, int) QueryCityDataForYears(string name, int year1, int
year2)
{
int population1 = 0, population2 = 0;
double area = 0;
Para obter mais informações sobre desconstruir tuplas com descartes, consulte Desconstruindo tuplas e outros
tipos.
O método Deconstruct de uma classe, estrutura ou interface também permite que você recupere e decomponha
um conjunto específico de dados de um objeto. Você poderá usar descartes quando estiver interessado em
trabalhar com apenas um subconjunto dos valores desconstruídos. O exemplo a seguir desconstrói um objeto
Person em quatro cadeias de caracteres (os nomes e sobrenomes, a cidade e o estado), mas descarta o
sobrenome e o estado.
using System;
public void Deconstruct(out string fname, out string mname, out string lname)
{
fname = FirstName;
mname = MiddleName;
lname = LastName;
}
// <Snippet1>
// Deconstruct the person object.
var (fName, _, city, _) = p;
Console.WriteLine($"Hello {fName} of {city}!");
// The example displays the following output:
// Hello John of Boston!
// </Snippet1>
}
}
// The example displays the following output:
// Hello John Adams of Boston, MA!
Para obter mais informações sobre desconstruir tipos definidos pelo usuário com descartes, consulte
Desconstruindo tuplas e outros tipos.
Correspondência de padrões com switch e is
O padrão de descarte pode ser usado na correspondência de padrões com as palavras-chave is e switch. Toda
expressão sempre corresponde ao padrão de descarte.
O exemplo a seguir define um método ProvidesFormatInfo que usa instruções is para determinar se um objeto
fornece uma implementação de IFormatProvider e testa se o objeto é null . Ele também usa o padrão de
descarte para manipular objetos não nulos de qualquer outro tipo.
using System;
using System.Globalization;
Um descarte autônomo
Você pode usar um descarte autônomo para indicar qualquer variável que você opte por ignorar. O exemplo a
seguir usa um descarte autônomo para ignorar o objeto Task retornado por uma operação assíncrona. Isso tem o
efeito de suprimir a exceção que a operação gera quando está prestes a ser concluída.
using System;
using System.Threading.Tasks;
Observe que _ também é um identificador válido. Quando usado fora de um contexto com suporte, _ não é
tratado como um descarte, mas como uma variável válida. Se um identificador chamado _ já está no escopo, o
uso de _ como um descarte autônomo pode resultar em:
A modificação acidental do valor da variável _ no escopo atribuindo a ela o valor do descarte pretendido.
Por exemplo:
Erro do compilador CS0136, "Um local ou um parâmetro denominado '_' não pode ser declarado neste
escopo porque esse nome é usado em um escopo delimitador de local para definir um local ou parâmetro."
Por exemplo:
public void DoSomething(int _)
{
var _ = GetValue(); // Error: cannot declare local _ when one is already in scope
}
// The example displays the following compiler error:
// error CS0136:
// A local or parameter named '_' cannot be declared in this scope
// because that name is used in an enclosing local scope
// to define a local or parameter
Consulte também
Desconstruindo tuplas e outros tipos
is palavra-chave
switch palavra-chave
Genéricos (Guia de Programação em C#)
30/10/2019 • 6 minutes to read • Edit Online
Os genéricos apresentam o conceito de parâmetros de tipo para o .NET Framework, o que possibilita criar classes e
métodos que adiam a especificação de um ou mais tipos até que a classe ou o método seja declarado e instanciado
pelo código do cliente. Por exemplo, usando um parâmetro de tipo genérico T , você pode escrever uma única
classe que outro código de cliente pode usar sem incorrer no custo ou risco de conversões de tempo de execução
ou operações de conversão boxing, conforme mostrado aqui:
Classes e métodos genéricos combinam reusabilidade, segurança de tipo e eficiência de forma que suas
contrapartes não genéricas não possam. Os genéricos são usados com mais frequência com coleções e com os
métodos que operam nelas. O namespace System.Collections.Generic contém várias classes de coleção baseadas
em genéricos. As coleções não genéricas, como ArrayList não são recomendadas e são mantidas para fins de
compatibilidade. Para saber mais, confira Genéricos no .NET.
É claro que você também pode criar tipos e métodos genéricos personalizados para fornecer suas próprias
soluções e padrões de design generalizados que sejam fortemente tipados e eficientes. O exemplo de código a
seguir mostra uma classe de lista vinculada genérica simples para fins de demonstração. (Na maioria dos casos,
você deve usar a classe List<T> fornecida pela .NET Framework biblioteca de classes em vez de criar a sua
própria.) O parâmetro de tipo T é usado em vários locais em que um tipo concreto normalmente seria usado para
indicar o tipo do item armazenado na lista. Ele é usado das seguintes maneiras:
Como o tipo de um parâmetro de método no método AddHead .
Como o tipo de retorno da propriedade Data na classe Node aninhada.
Como o tipo de data do membro particular na classe aninhada.
Observe que T está disponível para a classe Node aninhada. Quando GenericList<T> é instanciada com um tipo
concreto, por exemplo como um GenericList<int> , cada ocorrência de T será substituída por int .
// type parameter T in angle brackets
public class GenericList<T>
{
// The nested class is also generic on T.
private class Node
{
// T used in non-generic constructor.
public Node(T t)
{
next = null;
data = t;
}
// constructor
public GenericList()
{
head = null;
}
O exemplo de código a seguir mostra como o código cliente usa a classe GenericList<T> genérica para criar uma
lista de inteiros. Ao simplesmente alterar o argumento de tipo, o código a seguir poderia facilmente ser modificado
para criar listas de cadeias de caracteres ou qualquer outro tipo personalizado:
class TestGenericList
{
static void Main()
{
// int is the type argument
GenericList<int> list = new GenericList<int>();
Seções relacionadas
Parâmetros de tipo genérico
Restrições a parâmetros de tipo
Classes genéricas
Interfaces genéricas
Métodos genéricos
Delegados genéricos
Diferenças entre modelos C++ e genéricos C#
Genéricos e reflexão
Genéricos em tempo de execução
Especificação da linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#.
Consulte também
System.Collections.Generic
Guia de Programação em C#
Tipos
<typeparam>
<typeparamref>
Genéricos no .NET
Iterators
30/10/2019 • 10 minutes to read • Edit Online
Quase todos os programas que você escrever terão alguma necessidade de iterar em uma coleção. Você escreverá
um código que examina cada item em uma coleção.
Você também criará métodos de iterador que são métodos que produzem um iterador para os elementos dessa
classe. Eles podem ser usados para:
Executar uma ação em cada item em uma coleção.
Enumerar uma coleção personalizada.
Estender LINQ ou outras bibliotecas.
Criar um pipeline de dados em que os dados fluem com eficiência pelos métodos de iterador.
A linguagem C# fornece recursos para esses dois cenários. Este artigo fornece uma visão geral desses recursos.
Este tutorial tem várias etapas. Após cada etapa, você poderá executar o aplicativo e ver o progresso. Você também
pode exibir ou baixar o exemplo concluído deste tópico. Para obter instruções de download, consulte Exemplos e
tutoriais.
E isso é tudo. Para iterar em todo o conteúdo de uma coleção, a instrução foreach é tudo o que você precisa. No
entanto, a instrução foreach não é mágica. Ele se baseia em duas interfaces genéricas definidas na biblioteca do
.NET Core para gerar o código necessário para iterar uma coleção: IEnumerable<T> e IEnumerator<T> . Esse
mecanismo é explicado mais detalhadamente abaixo.
Essas duas interfaces também têm contrapartes não genéricas: IEnumerable e IEnumerator . As versões genéricas
são preferenciais para o código moderno.
O código acima mostra instruções yield return distintas para destacar o fato de que você pode usar várias
instruções yield return discretas em um método iterador. Você pode (e frequentemente o faz) usar outros
constructos de linguagem para simplificar o código de um método iterador. A definição do método abaixo produz
a mesma sequência exata de números:
Você não precisa determinar uma ou a outra. Você pode ter quantas instruções yield return forem necessárias
para atender as necessidades do seu método:
index = 100;
while (index < 110)
yield return index++;
}
Essa é a sintaxe básica. Vamos considerar um exemplo do mundo real em que você escreveria um método iterador.
Imagine que você está em um projeto de IoT e os sensores de dispositivo geram um enorme fluxo de dados. Para
ter uma noção dos dados, você pode escrever um método realiza a amostragem a cada N elementos de dados.
Esse pequeno método iterador resolve:
Há uma restrição importante em métodos de iterador: você não pode ter uma instrução return e uma instrução
yield return no mesmo método. O seguinte não compilará:
Essa restrição normalmente não é um problema. Você tem a opção de usar yield return em todo o método ou
separar o método original em vários métodos, alguns usando return e alguns usando yield return .
Você pode modificar um pouco o último método para usar yield return em todos os lugares:
var items = new int[] {100, 101, 102, 103, 104, 105, 106, 107, 108, 109 };
foreach (var item in items)
yield return item;
}
Às vezes, a resposta certa é dividir um método iterador em dois métodos diferentes. Um que usa return e outro
que usa yield return . Considere uma situação em que você talvez queira retornar uma coleção vazia ou os
primeiros cinco números ímpares, com base em um argumento booliano. Você poderia escrever isso como esses
dois métodos:
Observe os métodos acima. O primeiro usa a instrução return padrão para retornar uma coleção vazia ou o
iterador criado pelo segundo método. O segundo método usa a instrução yield return para criar a sequência
solicitada.
Aprofundamento em foreach
A instrução foreachse expande em uma expressão padrão que usa as interfaces IEnumerable<T> e
IEnumerator<T> para iterar em todos os elementos de uma coleção. Ela também minimiza os erros cometidos
pelos desenvolvedores por não gerenciarem os recursos adequadamente.
O compilador converte o loop foreach mostrado no primeiro exemplo em algo semelhante a esse constructo:
O constructo acima representa o código gerado pelo compilador C# da versão 5 e posterior. Antes da versão 5, a
variável item tinha um escopo diferente:
// C# versions 1 through 4:
IEnumerator<int> enumerator = collection.GetEnumerator();
int item = default(int);
while (enumerator.MoveNext())
{
item = enumerator.Current;
Console.WriteLine(item.ToString());
}
Isso foi alterado porque o comportamento anterior poderia levar a bugs sutis e difíceis de diagnosticar envolvendo
expressões lambda. Para saber mais sobre expressões lambda, confira o artigo sobre expressões lambda.
O código exato gerado pelo compilador é um pouco mais complicada e lida com situações em que o objeto
retornado por GetEnumerator() implementa a interface IDisposable . A expansão completa gera um código mais
semelhante a esse:
{
var enumerator = collection.GetEnumerator();
try
{
while (enumerator.MoveNext())
{
var item = enumerator.Current;
Console.WriteLine(item.ToString());
}
} finally
{
// dispose of enumerator.
}
}
A maneira na qual o enumerador é descartado depende das características do tipo de enumerator . Em geral, a
cláusula finally expande para:
finally
{
(enumerator as IDisposable)?.Dispose();
}
No entanto, se o tipo de enumerator é um tipo lacrado e não há nenhuma conversão implícita do tipo de
enumerator para IDisposable , a cláusula finally se expande para um bloco vazio:
finally
{
}
Se houver uma conversão implícita do tipo de enumerator para IDisposable e enumerator for um tipo de valor
que não aceita valores nulos, a cláusula finally se expandirá para:
finally
{
((IDisposable)enumerator).Dispose();
}
Felizmente, você não precisa se lembrar de todos esses detalhes. A instrução foreach trata todas essas nuances
para você. O compilador gerará o código correto para qualquer um desses constructos.
Introdução a Delegados
30/10/2019 • 4 minutes to read • Edit Online
Os delegados fornecem um mecanismo de associação tardia no .NET. Associação tardia significa que você cria
um algoritmo em que o chamador também fornece pelo menos um método que implementa a parte do
algoritmo.
Por exemplo, considere classificar uma lista de estrelas em um aplicativo de astronomia. Você pode optar por
classificar as estrelas por sua distância da terra ou a magnitude da estrela ou seu brilho percebido.
Em todos esses casos, o método Sort() faz essencialmente a mesma coisa: organiza os itens na lista com base em
alguma comparação. O código que compara duas estrelas é diferente para cada uma das ordenações de
classificação.
Esses tipos de soluções foram usados no software por meio século. O conceito de delegado de linguagem C#
fornece suporte de linguagem de primeira classe e segurança de tipos em torno do conceito.
Como você verá posteriormente nesta série, o código C# que você escreve para algoritmos como esse é seguro
quanto ao tipo e aproveita a linguagem e o compilador para garantir que os tipos correspondem aos argumentos
e tipos de retorno.
Anterior
Este artigo abordará as classes do .NET Framework que dão suporte a delegados e como eles são mapeados para
a palavra-chave delegate .
O compilador gera uma classe, derivada de System.Delegate , que corresponde à assinatura usada (nesse caso, um
método que retorna um inteiro e tem dois argumentos). O tipo do delegado é Comparison . O tipo delegado
Comparison é um tipo genérico. Para obter detalhes sobre os genéricos, consulte aqui.
Observe que a sintaxe pode aparecer como se estivesse declarando uma variável, mas na verdade está declarando
um tipo. Você pode definir tipos de delegado dentro de classes, diretamente dentro de namespaces ou até mesmo
no namespace global.
NOTE
Declarar tipos de delegado (ou outros tipos) diretamente no namespace global não é recomendado.
O compilador também gera manipuladores de adição e remoção para esse novo tipo de forma que os clientes
dessa classe podem adicionar e remover métodos de uma lista de invocação de instância. O compilador imporá
que a assinatura do método que está sendo adicionado ou removido corresponda à assinatura usada ao declarar o
método.
O tipo da variável é Comparison<T> , o tipo de delegado definido anteriormente. O nome da variável é comparator .
Esse snippet de código acima declarou uma variável de membro dentro de uma classe. Você também pode
declarar variáveis de delegado que são variáveis locais ou argumentos para métodos.
Invocando delegados
Você invoca os métodos que estão na lista de invocação de um delegado chamando esse delegado. Dentro do
método Sort() , o código chamará o método de comparação para determinar em qual ordem posicionar objetos:
Na linha acima, o código invoca o método anexado ao delegado. Você trata a variável como um nome de método
e a invoca usando a sintaxe de chamada de método normal.
Essa linha de código faz uma suposição não segura: não há garantia de que um destino foi adicionado ao
delegado. Se nenhum destino tiver sido anexado, a linha acima fará com que um NullReferenceException seja
lançado. As expressões usadas para resolver esse problema são mais complicadas do que uma simples verificação
de null e são abordadas posteriormente nesta série.
O método é declarado como um método particular. Tudo bem. Você pode não desejar que esse método seja parte
da sua interface pública. Ele ainda pode ser usado como o método de comparação ao anexar a um delegado. O
código de chamada terá esse método anexado à lista de destino do objeto de delegado e pode acessá-lo por meio
do delegado.
Você cria essa relação passando esse método para o método List.Sort() :
phrases.Sort(CompareLength);
Observe que o nome do método é usado, sem parênteses. Usar o método como um argumento informa ao
compilador para converter a referência de método em uma referência que pode ser usada como um destino de
invocação do delegado e anexar esse método como um destino de invocação.
Você também poderia ter sido explícito declarando uma variável do tipo Comparison<string> e fazendo uma
atribuição:
Comparison<string> comparer = CompareLength;
phrases.Sort(comparer);
Em utilizações em que o método que está sendo usado como um destinos de delegado é um método pequeno, é
comum usar a sintaxe da expressão lambda para executar a atribuição:
O uso de expressões lambda para destinos de delegado será abordado em uma seção posterior.
O exemplo de Sort() normalmente anexa um único método de destino ao delegado. No entanto, objetos
delegados dão suporte a listas de invocação que têm vários métodos de destino anexados a um objeto de
delegado.
Anterior
No artigo anterior, você viu como criar tipos de delegado específicos usando a palavra-chave delegate .
A classe de delegado abstrata fornece a infraestrutura para a invocação e acoplamento fraco. Os tipos de delegado
concretos se tornam muito mais úteis adotando e impondo a segurança de tipos para os métodos que são
adicionados à lista de invocação para um objeto delegado. Quando você usa a palavra-chave delegate e define
um tipo de delegado concreto, o compilador gera esses métodos.
Na prática, isso poderia levar à criação de novos tipos de delegado sempre que precisar de uma assinatura de
método diferente. Esse trabalho pode se tornar entediante depois de um tempo. Cada novo recurso exige novos
tipos de delegado.
Felizmente, isso não é necessário. O .NET Core Framework contém vários tipos que podem ser reutilizados
sempre que você precisar de tipos de delegado. Essas são definições genéricas para que você possa declarar
personalizações quando precisar de novas declarações de método.
O primeiro desses tipos é o tipo Action e diversas variações:
O modificador out no argumento de tipo genérico de resultado é abordado neste artigo sobre covariância.
Há variações do delegado Func com até 16 argumentos de entrada como
Func<T1,T2,T3,T4,T5,T6,T7,T8,T9,T10,T11,T12,T13,T14,T15,T16,TResult>. O tipo do resultado é sempre o último
parâmetro de tipo em todas as declarações Func , por convenção.
Use um dos tipos Func para qualquer tipo de delegado que retorna um valor.
Também há um Predicate<T> especializado tipo para um delegado que retorna um teste em um único valor:
public delegate bool Predicate<in T>(T obj);
Você pode observar que para qualquer tipo Predicate , há um tipo Func estruturalmente equivalente, por
exemplo:
Você pode pensar que esses dois tipos são equivalentes. Eles não são. Essas duas variáveis não podem ser usadas
alternadamente. Uma variável de um tipo não pode ser atribuída o outro tipo. O sistema de tipo do C# usa os
nomes dos tipos definidos, não a estrutura.
Todas essas definições de tipo de delegado na Biblioteca do .NET Core devem significar que você não precisa
definir um novo tipo de delegado para qualquer novo recurso criado que exige delegados. Essas definições
genéricas devem fornecer todos os tipos de delegado necessários na maioria das situações. Você pode
simplesmente instanciar um desses tipos com os parâmetros de tipo necessários. No caso de algoritmos que
podem ser tornados genéricos, esses delegados podem ser usados como tipos genéricos.
Isso deve economizar tempo e minimizar o número de novos tipos de que você precisa criar a fim de trabalhar
com delegados.
No próximo artigo, você verá vários padrões comuns para trabalhar com delegados na prática.
Avançar
Padrões Comuns para Delegados
04/11/2019 • 15 minutes to read • Edit Online
Anterior
Os delegados fornecem um mecanismo que permite designs de software que envolvem acoplamento mínimo
entre os componentes.
Um exemplo excelente desse tipo de design é o LINQ. O padrão de expressão de consulta LINQ se baseia em
delegados para todos os seus recursos. Considere este exemplo simples:
Isso filtra a sequência de números para somente aqueles com valor menor que 10. O método Where usa um
delegado que determina quais elementos de uma sequência são passados no filtro. Quando cria uma consulta
LINQ, você fornece a implementação do delegado para essa finalidade específica.
O protótipo para o método Where é:
Este exemplo é repetido com todos os métodos que fazem parte do LINQ. Todos eles contam com delegados
para o código que gerencia a consulta específica. Trata-se de um padrão de design de API muito eficiente para se
aprender e entender.
Este exemplo simples ilustra como delegados requerem muito pouco acoplamento entre componentes. Você não
precisa criar uma classe que deriva de uma classe base específica. Você não precisa implementar uma interface
específica. O único requisito é fornecer a implementação de um método que é fundamental para a tarefa em
questão.
A classe estática acima é a coisa mais simples que pode funcionar. Precisamos escrever a única implementação
do método que grava mensagens no console:
Por fim, você precisa conectar o delegado anexando-o ao delegado WriteMessage declarado no agente:
Logger.WriteMessage += LoggingMethods.LogToConsole;
Práticas
Até agora, nossa amostra é bastante simples, mas ainda demonstra algumas das diretrizes importantes para
designs que envolvem delegados.
Usar os tipos de delegados definidos no Core Framework torna mais fácil para os usuários trabalhem com os
delegados. Você não precisa definir novos tipos e os desenvolvedores que usam sua biblioteca não precisam
aprender novos tipos de delegados especializadas.
As interfaces usadas são tão mínimas e flexíveis quanto possível: para criar um novo agente de saída, você
precisa criar um método. O método pode ser um método estático ou um método de instância. Ele pode ter
qualquer acesso.
Saída de formatação
Vamos fazer esta primeira versão um pouco mais robusta e, então, começar a criar outros mecanismos de
registro em log.
Em seguida, vamos adicionar alguns argumentos para o método LogMessage() para que sua classe de log crie
mensagens mais estruturadas:
public enum Severity
{
Verbose,
Trace,
Information,
Warning,
Error,
Critical
}
Em seguida, vamos usar aquele argumento Severity para filtrar as mensagens que são enviadas para o log de
saída.
Práticas
Você adicionou novos recursos à infraestrutura de registro em log. Como somente o componente do agente está
acoplado de forma muito flexível a qualquer mecanismo de saída, esses novos recursos podem ser adicionados
sem afetar o código que implementa o delegado do agente.
Conforme prosseguir com sua criação, você verá mais exemplos de como esse acoplamento flexível permite
maior versatilidade para atualizar partes do site sem alterar outros locais. De fato, em um aplicativo maior, as
classes de saída do agente podem estar em um assembly diferente e nem mesmo precisar ser recriadas.
Após ter criado essa classe, você pode instanciá-la e ela anexa o método LogMessage ao componente do agente:
Os dois não são mutuamente exclusivos. Você pode anexar os dois métodos de log e gerar mensagens para o
console e um arquivo:
Posteriormente, mesmo no mesmo aplicativo, você pode remover um dos delegados sem problemas para o
sistema:
Logger.WriteMessage -= LoggingMethods.LogToConsole;
Práticas
Agora, você adicionou um segundo manipulador de saída para o subsistema de registro em log. Este precisa de
um pouco mais infraestrutura para dar suporte ao sistema de arquivos corretamente. O delegado um método de
instância. Ele também é um método particular. Não é necessário ter mais acessibilidade porque a infraestrutura
de delegados pode conectar os delegados.
Em segundo lugar, o design baseado em delegados permite vários métodos de saída sem nenhum código extra.
Não é necessário criar nenhuma infraestrutura adicional para dar suporte a vários métodos de saída. Eles
simplesmente se tornam outro método na lista de invocação.
Dedique atenção especial ao código no método de saída do registro em log de arquivos. Ele é codificado para
garantir que não gere nenhuma exceção. Embora isso nem sempre seja estritamente necessário, geralmente é
uma boa prática. Se um dos métodos de delegado gerar uma exceção, os delegados restantes que fazem parte
da invocação não serão invocados.
Como uma última observação, o agente de arquivos deve gerenciar seus recursos abrindo e fechando o arquivo
em cada mensagem de log. Você pode optar por manter o arquivo aberto e implementar IDisposable para
fechar o arquivo quando terminar. Cada método tem suas vantagens e desvantagens. Ambos criam um pouco
mais de acoplamento entre as classes.
Nenhum código na classe de agente precisaria ser atualizado para dar suporte a um dos cenários.
O operador condicional nulo ( ?. ) entra em curto-circuito quando o operando esquerdo ( WriteMessage nesse
caso) for nulo, o que significa que não é feita nenhuma tentativa de registrar uma mensagem.
Você não encontrará o método Invoke() listado na documentação de System.Delegate ou
System.MulticastDelegate . O compilador gera um método Invoke fortemente tipado para qualquer tipo de
delegado declarado. Neste exemplo, isso significa que Invoke usa um único argumento string e tem um tipo
de retorno nulo.
Anterior
Eventos são, assim como delegados, um mecanismo de associação tardia. De fato, os eventos são criados com
base no suporte de linguagem para delegados.
Os eventos são uma forma de um objeto difundir (para todos os componentes interessados do sistema) que algo
aconteceu. Qualquer outro componente pode assinar ao evento e ser notificado quando um evento for gerado.
Provavelmente, você usou eventos em alguma parte de sua programação. Muitos sistemas gráficos têm um
modelo de evento para informar a interação do usuário. Esses eventos informariam movimentos do mouse,
pressionamentos de botão e interações semelhantes. Esse é um dos cenários mais comuns, mas certamento não o
único cenário em que eventos são usados.
Você pode definir os eventos que devem ser gerados para suas classes. Uma consideração importante ao
trabalhar com eventos é que pode não haver nenhum objeto registrado para um determinado evento. Você deve
escrever seu código de modo que ele não gere eventos quando nenhum ouvinte estiver configurado.
Assinar um evento também cria um acoplamento entre dois objetos (a origem do evento e o coletor do evento).
Você precisa garantir que o coletor do evento cancele a assinatura da origem do evento quando não houver mais
interesse nos eventos.
O tipo de evento ( EventHandler<FileListArgs> neste exemplo) deve ser um tipo delegado. Há uma série de
convenções que você deve seguir ao declarar um evento. Normalmente, o tipo de delegado do evento tem um
retorno nulo. Declarações de evento devem ser um verbo ou uma frase verbal. Use conjugação anteriores quando
o evento relatar algo que aconteceu. Use um tempo verbal presente (por exemplo, Closing ) para informar algo
que está prestes a ocorrer. Frequentemente, usar o tempo presente indica que sua classe dá suporte a algum tipo
de comportamento de personalização. Um dos cenários mais comuns é dar suporte ao cancelamento. Por
exemplo, um evento Closing pode incluir um argumento que indicaria se a operação de encerramento deve
continuar ou não. Outros cenários podem permitir que os chamadores modifiquem o comportamento atualizando
propriedades dos argumentos do evento. Você pode acionar um evento para indicar uma próxima ação proposta
que um algoritmo usará. O manipulador de eventos pode forçar uma ação diferente modificando as propriedades
do argumento do evento.
Quando quiser acionar o evento, você pode chamar os manipuladores de eventos usando a sintaxe de invocação
de delegado:
Conforme discutido na seção sobre delegados, o operador ?. torna fácil garantir que você não tente acionar o
evento quando não houver nenhum assinante do evento.
Assine um evento usando o operador += :
fileLister.Progress += onProgress;
O método Handler normalmente tem o prefixo ' on ' seguido pelo nome do evento, como mostrado acima.
Cancele a assinatura usando o operador -= :
fileLister.Progress -= onProgress;
É importante observar que eu declarei um variável local para a expressão que representa o manipulador de
eventos. Isso garante que o cancelamento da assinatura remova o manipulador. Se, em vez disso, você tiver usado
o corpo da expressão lambda, você estará tentando remover um manipulador que nunca foi anexado, o que não
faz nada.
No próximo artigo, você aprenderá mais sobre padrões de evento típicos e diferentes variações deste exemplo.
Avançar
Padrões de evento .NET padrão
30/10/2019 • 15 minutes to read • Edit Online
Anterior
Os eventos do .NET geralmente seguem alguns padrões conhecidos. Adotar esses padrões significa que os
desenvolvedores podem aproveitar o conhecimento desses padrões, que podem ser aplicados a qualquer
programa de evento do .NET.
Vamos analisar esses padrões para que você obtenha todo o conhecimento que precisa a fim de criar origens do
evento padrão e também assinar e processar eventos padrão em seu código.
O tipo de retorno é nulo. Os eventos são baseados em delegados e são delegados multicast. Isso dá suporte a
vários assinantes de qualquer origem do evento. O único valor retornado de um método não ajusta a escala para
vários assinantes do evento. Qual valor retornado a origem do evento vê depois de gerar um evento? Neste artigo,
você verá como criar protocolos de evento que oferecem suporte a assinantes de evento que relatam informações
para a origem do evento.
A lista de argumentos contém dois argumentos: o remetente e os argumentos do evento. O tipo de tempo de
compilação de sender é System.Object , mas é provável que você conheça um tipo mais derivado que sempre
estaria correto. Por convenção, use object .
O segundo argumento normalmente tem sido um tipo derivado de System.EventArgs . (Você verá na próxima
seção que essa Convenção não é mais imposta.) Se o seu tipo de evento não precisar de nenhum argumento
adicional, você ainda fornecerá ambos os argumentos. Há um valor especial, o EventArgs.Empty , que você deve
usar para indicar que o evento não contém nenhuma informação adicional.
Vamos criar uma classe que lista os arquivos em um diretório ou em qualquer um de seus subdiretórios, que
seguem um padrão. Esse componente aciona um evento para cada arquivo encontrado que corresponde ao
padrão.
O uso de um modelo de evento fornece algumas vantagens de design. Você pode criar vários ouvintes de eventos
que realizam ações diferentes quando um arquivo procurado é encontrado. A combinação de diferentes ouvintes
pode criar algoritmos mais robustos.
Aqui está a declaração de argumento de evento inicial para localizar um arquivo pesquisado:
Isso é semelhante à declaração de um campo público, o que parece ser uma prática ruim orientada a objetos. Você
deseja proteger o acesso a dados por meio de propriedades ou métodos. Embora isso possa parecer uma prática
ruim, o código gerado pelo compilador cria wrappers para que os objetos de evento só possam ser acessados de
maneiras seguras. As únicas operações disponíveis em um evento semelhante a campo são adicionar
manipulador:
fileLister.FileFound += onFileFound;
e remover manipulador:
fileLister.FileFound -= onFileFound;
Observe que há uma variável local para o manipulador. Se você usou o corpo de lambda, a operação remover não
funcionará corretamente. Ela seria uma instância diferente do delegado e silenciosamente não faria nada.
O código fora da classe não pode acionar o evento nem executar outras operações.
Valor retornados de assinantes de evento
Sua versão simples está funcionando bem. Vamos adicionar outro recurso: cancelamento.
Quando você acionar o evento encontrado, os ouvintes devem ser capazes de parar o processamento, se esse
arquivo for aquele que era procurado.
Os manipuladores de eventos não retornam um valor, por isso você precisa comunicar isso de outra forma. O
padrão de evento usa o objeto EventArgs para incluir campos que os assinantes de evento podem usar para
comunicar o cancelamento.
Há dois padrões diferentes que podem ser usados, com base na semântica do contrato de cancelamento. Em
ambos os casos, você adicionará um campo booliano no EventArguments para o evento de arquivo encontrado.
Um padrão permitiria a qualquer assinante cancelar a operação. Para esse padrão, o novo campo é inicializado
para false . Qualquer assinante pode alterá-lo para true . Depois que todos os assinantes viram o evento
acionado, o componente FileSearcher examina o valor booliano e toma uma ação.
O segundo padrão só cancelaria a operação se todos os assinantes quisessem que a operação fosse cancelada.
Nesse padrão, o novo campo é inicializado para indicar que a operação deve ser cancelada e qualquer assinante
poderia alterá-lo para indicar que a operação deve continuar. Depois que todos os assinantes viram o evento
acionado, o componente FileSearcher examina o booliano e toma uma ação. Há uma etapa adicional nesse padrão:
o componente precisa saber se algum assinante viu o evento. Se não houver nenhum assinante, o campo indicaria
incorretamente um cancelamento.
Vamos implementar a primeira versão deste exemplo. Você precisa adicionar um campo booliano chamado
CancelRequested ao tipo FileFoundArgs :
Este novo campo é inicializado automaticamente para false , o valor padrão para um campo booliano, para que
você não cancele acidentalmente. A única alteração adicional no componente é verificar o sinalizador depois de
acionar o evento, para ver se qualquer um dos assinantes solicitou um cancelamento:
Uma vantagem desse padrão é que ele não é uma alteração significativa. Nenhum dos assinantes solicitou um
cancelamento antes e ainda não o fizeram. Nenhuma parte do código de assinante precisa ser atualizado, a menos
que eles queiram dar suporte ao novo protocolo de cancelamento. Ele é acoplado de forma bem livre.
Vamos atualizar o assinante para que ele solicite um cancelamento, depois de encontrar o primeiro executável:
EventHandler<FileFoundArgs> onFileFound = (sender, eventArgs) =>
{
Console.WriteLine(eventArgs.FoundFile);
eventArgs.CancelRequested = true;
};
Novamente, você pode seguir as recomendações para criar um tipo de referência imutável para os argumentos do
evento.
Em seguida, defina o evento. Desta vez, você usará uma sintaxe diferente. Além de usar a sintaxe de campo, você
pode explicitamente criar a propriedade com os manipuladores adicionar e remover. Nesta amostra, você não
precisará de código extra nos manipuladores, mas será mostrado como criá-los.
De muitas formas, o código que você escreverá aqui é bem parecido com o código que o compilador gera para as
definições de evento de campo vistas anteriormente. Você cria o evento usando uma sintaxe muito parecida àquela
utilizada para propriedades. Observe que os manipuladores têm nomes diferentes: add e remove . Eles são
chamados para assinar o evento ou cancelar a inscrição do evento. Observe que você também deve declarar um
campo de suporte particular para armazenar a variável de evento. Ele é inicializado como null.
Em seguida, vamos adicionar a sobrecarga do método Search que percorre os subdiretórios e aciona os dois
eventos. A maneira mais fácil de fazer isso é usar um argumento padrão para especificar que você deseja
pesquisar todas as pastas:
public void Search(string directory, string searchPattern, bool searchSubDirs = false)
{
if (searchSubDirs)
{
var allDirectories = Directory.GetDirectories(directory, "*.*", SearchOption.AllDirectories);
var completedDirs = 0;
var totalDirs = allDirectories.Length + 1;
foreach (var dir in allDirectories)
{
directoryChanged?.Invoke(this,
new SearchDirectoryArgs(dir, totalDirs, completedDirs++));
// Search 'dir' and its subdirectories for files that match the search pattern:
SearchDirectory(dir, searchPattern);
}
// Include the Current Directory:
directoryChanged?.Invoke(this,
new SearchDirectoryArgs(directory, totalDirs, completedDirs++));
SearchDirectory(directory, searchPattern);
}
else
{
SearchDirectory(directory, searchPattern);
}
}
Neste momento, você pode executar o aplicativo, chamando a sobrecarga para pesquisar todos os subdiretórios.
Não há nenhum assinante no novo evento ChangeDirectory , mas o uso da expressão ?.Invoke() garante que isso
funcione corretamente.
Vamos adicionar um manipulador para escrever uma linha que mostra o andamento na janela do console.
Você viu os padrões que são seguidos em todo o ecossistema do .NET. Ao aprender esses padrões e convenções,
você escreverá expressões idiomáticas de C# e .NET rapidamente.
Em seguida, você verá algumas alterações nesses padrões na versão mais recente do .NET.
Avançar
O padrão de eventos atualizado do .NET Core Event
30/10/2019 • 7 minutes to read • Edit Online
Anterior
O artigo anterior abordou os padrões de eventos mais comuns. O .NET Core tem um padrão mais flexível. Nesta
versão, a definição EventHandler<TEventArgs> não tem a restrição de que TEventArgs deve ser uma classe
derivada de System.EventArgs .
Isso aumenta a flexibilidade para você e é compatível com versões anteriores. Vamos começar com a flexibilidade.
A classe System.EventArgs introduz um método: MemberwiseClone() , que cria uma cópia superficial do objeto.
Esse método deve usar a reflexão para implementar sua funcionalidade para qualquer classe derivada de
EventArgs . Essa funcionalidade é mais fácil de criar em uma classe derivada específica. Na prática, isso significa
que a derivação de System.EventArgs é uma restrição que limita seus designs, mas não oferece nenhum benefício
adicional. Na verdade, você pode alterar as definições de FileFoundArgs e SearchDirectoryArgs para que eles não
derivem de EventArgs . O programa funcionará exatamente da mesma forma.
Você também pode alterar o SearchDirectoryArgs para um struct, se você fizer mais uma alteração:
A alteração adicional é chamar o construtor sem parâmetro antes de inserir o construtor que inicializa todos os
campos. Sem esse acréscimo, as regras de C# informariam que as propriedades estão sendo acessadas antes de
terem sido atribuídas.
Você não deve alterar o FileFoundArgs de uma classe (tipo de referência) para um struct (tipo de valor). Isso
ocorre porque o protocolo para manipular cancelamentos exige que os argumentos do evento sejam passados
por referência. Se você fizesse a mesma alteração, a classe de pesquisa de arquivo nunca observaria as alterações
feitas por qualquer um dos assinantes do evento. Uma nova cópia da estrutura seria usada para cada assinante e
essa cópia seria uma cópia diferente daquela vista pelo objeto de pesquisa de arquivo.
Em seguida, vamos considerar como essa alteração pode ser compatível com versões anteriores. A remoção da
restrição não afeta nenhum código existente. Qualquer tipo de argumento de evento existente ainda deriva de
System.EventArgs . A compatibilidade com versões anteriores é um dos principais motivos pelos quais eles vão
continuar derivando de System.EventArgs . Assinantes de eventos existentes serão assinantes de um evento que
seguiu o padrão clássico.
Seguindo uma lógica semelhante, qualquer tipo de argumento de evento criado agora não teria assinantes nas
bases de código existentes. Novos tipos de evento que não derivam de System.EventArgs não quebram essas
bases de código.
Eventos com assinantes assíncronos
Você tem um último padrão para aprender: como escrever corretamente os assinantes do evento que chamam o
código assíncrono. O desafio é descrito no artigo em async e await. Métodos assíncronos podem ter um tipo de
retorno nulo, mas isso não é recomendável. Quando seu código de assinante de evento chama um método
assíncrono, você não terá outra escolha além de criar um método async void . A assinatura do manipulador de
eventos o exige.
Você precisa conciliar essas diretrizes opostas. De alguma forma, você precisa criar um método async void
seguro. As noções básicas do padrão que você precisa implementar estão abaixo:
Primeiro, observe que o manipulador está marcado como um manipulador assíncrono. Como está sendo
atribuído a um tipo de delegado de manipulador de eventos, ele terá um tipo de retorno nulo. Isso significa que
você deve seguir o padrão mostrado no manipulador e não permitir que qualquer exceção seja gerada fora do
contexto do manipulador assíncrono. Como ele não retorna uma tarefa, não há nenhuma tarefa que pode relatar o
erro entrando no estado de falha. Como o método é assíncrono, ele não pode simplesmente gerar a exceção. (O
método de chamada continuou a execução porque está async .) O comportamento real do tempo de execução
será definido de forma diferente para ambientes diferentes. Ele pode encerrar o thread ou o processo que possui o
thread ou deixar o processo em um estado indeterminado. Todos esses resultados potenciais são altamente
indesejáveis.
É por isso que você deve encapsular a instrução await para a tarefa assíncrona em seu próprio bloco de teste. Se
isso causar uma tarefa com falha, você pode registrar o erro em log. Se for um erro do qual não é possível
recuperar o aplicativo, você pode sair do programa rápida e normalmente
Essas são as principais atualizações do padrão de eventos do .NET. Você verá muitos exemplos das versões
anteriores nas bibliotecas com que trabalhar. No entanto, você também precisa compreender quais são os padrões
mais recentes.
O próximo artigo desta série ajuda a distinguir entre o uso de delegates e de events em seus designs. Eles são
conceitos similares e artigo o ajudará a tomar a melhor decisão para seus programas.
Avançar
Distinção entre Delegados e Eventos
31/10/2019 • 6 minutes to read • Edit Online
Anterior
Desenvolvedores que são novos na plataforma .NET Core geralmente têm dificuldades para decidir entre um
design baseado em delegates e um design baseado em events . Este é um conceito difícil, porque os recursos das
duas linguagens são muito semelhantes. De fato, os eventos são criados usando o suporte de linguagem para
delegados.
Ambas oferecem um cenário de associação tardia: elas permitem cenários em que um componente se comunica
chamando um método que é conhecido somente em runtime. Ambas dão suporte a métodos de assinante único e
vários assinantes. Você pode ver esse suporte ser chamado de singlecast e multicast. Ambas dão suporte a uma
sintaxe semelhante para adicionar e remover manipuladores. Por fim, acionar um evento e chamar um delegado
usam exatamente a mesma sintaxe de chamada de método. As duas até mesmo dão suporte à mesma sintaxe de
método Invoke() para uso com o operador ?. .
Com todas essas semelhanças, é fácil de ter problemas para determinar quando usar qual.
Avalie cuidadosamente
As considerações acima não são regras rígidas e óbvias. Em vez disso, elas são diretrizes que podem ajudá-lo a
decidir qual opção é melhor para seu uso específico. Como elas são semelhantes, você pode até mesmo fazer
protótipos das suas e considerar com qual seria mais natural trabalhar. Ambas lidam bem com cenários de
associação tardia. Use a que comunica melhor o seu design.
LINQ (Consulta Integrada à Linguagem)
23/10/2019 • 6 minutes to read • Edit Online
class LINQQueryExpressions
{
static void Main()
{
Próximas etapas
Para obter mais detalhes sobre o LINQ, comece se familiarizando com alguns conceitos básicos em
Noções básicas sobre expressões de consulta, e, em seguida, leia a documentação para a tecnologia LINQ
na qual você está interessado:
Documentos XML: LINQ to XML
ADO.NET Entity Framework: LINQ to entities
Coleções do .NET, arquivos, cadeias de caracteres e assim por diante: LINQ to objects
Para saber mais sobre o LINQ, consulte LINQ em C#.
Para começar a trabalhar com o LINQ em C#, consulte o tutorial Trabalhando com LINQ.
Noções básicas sobre expressões de consulta
23/10/2019 • 22 minutes to read • Edit Online
IEnumerable<int> highScoresQuery =
from score in scores
where score > 80
orderby score descending
select score;
Recuperar uma sequência de elementos, como no exemplo anterior, mas transformá-los em um novo tipo
de objeto. Por exemplo, uma consulta pode recuperar apenas os sobrenomes de determinados registros de
cliente em uma fonte de dados. Ou pode recuperar o registro completo e, em seguida, usá-lo para construir
outro tipo de objeto na memória ou até mesmo dados XML antes de gerar a sequência de resultados final.
O exemplo a seguir mostra uma projeção de um int para um string . Observe o novo tipo de
highScoresQuery .
IEnumerable<string> highScoresQuery2 =
from score in scores
where score > 80
orderby score descending
select $"The score is {score}";
int highScoreCount =
(from score in scores
where score > 80
select score)
.Count();
IEnumerable<int> highScoresQuery3 =
from score in scores
where score > 80
select score;
No exemplo anterior, a consulta é executada na chamada para Count , pois Count deve iterar os resultados para
determinar o número de elementos retornados por highScoresQuery .
O exemplo de código a seguir mostra uma expressão de consulta simples com uma fonte de dados, uma cláusula
de filtragem, uma cláusula de ordenação e nenhuma transformação dos elementos de origem. A cláusula select
termina a consulta.
static void Main()
{
// Data source.
int[] scores = { 90, 71, 82, 93, 75, 82 };
// Query Expression.
IEnumerable<int> scoreQuery = //query variable
from score in scores //required
where score > 80 // optional
orderby score descending // optional
select score; //must end with select or group
No exemplo anterior, scoreQuery é uma variável de consulta, o que às vezes é chamado apenas de uma consulta.
A variável de consulta não armazena nenhum dado de resultado real, que é produzido no loop foreach . E quando
instrução foreach é executada, os resultados da consulta não são retornados pela variável de consulta
scoreQuery . Em vez disso, eles são retornados pela variável de iteração testScore . A variável scoreQuery pode
ser iterada em um segundo loop foreach . Ele produzirá os mesmos resultados contanto que nem ele nem a fonte
de dados tenham sido modificados.
Uma variável de consulta pode armazenar uma consulta que é expressada na sintaxe de consulta ou na sintaxe de
método ou uma combinação das duas. Nos exemplos a seguir, queryMajorCities e queryMajorCities2 são
variáveis de consulta:
//Query syntax
IEnumerable<City> queryMajorCities =
from city in cities
where city.Population > 100000
select city;
// Method-based syntax
IEnumerable<City> queryMajorCities2 = cities.Where(c => c.Population > 100000);
Por outro lado, os dois exemplos a seguir mostram variáveis que não são variáveis de consulta, embora sejam
inicializadas com uma consulta. Elas não são variáveis de consulta porque armazenam resultados:
int highestScore =
(from score in scores
select score)
.Max();
List<City> largeCitiesList =
(from country in countries
from city in country.Cities
where city.Population > 10000
select city)
.ToList();
Para obter mais informações sobre as diferentes maneiras de expressar consultas, consulte Sintaxe de consulta e
sintaxe de método em LINQ.
Tipagem explícita e implícita de variáveis de consulta
Esta documentação normalmente fornece o tipo explícito da variável de consulta para mostrar a relação de tipo
entre a variável de consulta e a cláusula select. No entanto, você também pode usar a palavra-chave var para
instruir o compilador a inferir o tipo de uma variável de consulta (ou qualquer outra variável local) em tempo de
compilação. Por exemplo, o exemplo de consulta que foi mostrado anteriormente neste tópico também pode ser
expressado usando a tipagem implícita:
Para obter mais informações, consulte Variáveis locais de tipo implícito e Relacionamentos de tipo em operações
de consulta LINQ.
Iniciando uma expressão de consulta
Uma expressão de consulta deve começar com uma cláusula from . Especifica uma fonte de dados junto com uma
variável de intervalo. A variável de intervalo representa cada elemento sucessivo na sequência de origem como a
sequência de origem que está sendo percorrida. A variável de intervalo é fortemente tipada com base no tipo dos
elementos na fonte de dados. No exemplo a seguir, como countries é uma matriz de objetos Country , a variável
de intervalo também é tipada como Country . Como a variável de intervalo é fortemente tipada, você pode usar o
operador ponto para acessar todos os membros disponíveis do tipo.
IEnumerable<Country> countryAreaQuery =
from country in countries
where country.Area > 500000 //sq km
select country;
A variável de intervalo está no escopo até a consulta ser encerrada com ponto e vírgula ou com uma cláusula
continuation.
Uma expressão de consulta pode conter várias cláusulas from . Use cláusulas from adicionais quando cada
elemento na sequência de origem for uma coleção em si ou contiver uma coleção. Por exemplo, suponha que você
tem uma coleção de objetos Country e cada um dos quais contém uma coleção de objetos City chamada
Cities . Para consular os objetos City em cada Country , use duas cláusulas from como mostrado aqui:
IEnumerable<City> cityQuery =
from country in countries
from city in country.Cities
where city.Population > 10000
select city;
var queryCountryGroups =
from country in countries
group country by country.Name[0];
IEnumerable<Country> sortedQuery =
from country in countries
orderby country.Area
select country;
A cláusula select pode ser usada para transformar dados de origem em sequências de novos tipos. Essa
transformação também é chamada de projeção. No exemplo a seguir, a cláusula select projeta uma sequência
de tipos anônimos que contém apenas um subconjunto dos campos no elemento original. Observe que os novos
objetos são inicializados usando um inicializador de objeto.
// Here var is required because the query
// produces an anonymous type.
var queryNameAndPop =
from country in countries
select new { Name = country.Name, Pop = country.Population };
Para obter mais informações sobre todas as maneiras que uma cláusula select pode ser usada para transformar
os dados de origem, consulte Cláusula select.
Continuações com "into"
Você pode usar a palavra-chave into em uma cláusula select ou group para criar um identificador temporário
que armazena uma consulta. Faça isso quando precisar executar operações de consulta adicionais em uma
consulta após a operação de agrupamento ou seleção. No exemplo a seguir countries são agrupados de acordo
com a população em intervalos de 10 milhões. Depois que esses grupos são criados, cláusulas adicionais filtram
alguns grupos e, em seguida, classificam os grupos em ordem crescente. Para executar essas operações adicionais,
a continuação representada por countryGroup é necessária.
IEnumerable<City> queryCityPop =
from city in cities
where city.Population < 200000 && city.Population > 100000
select city;
A palavra-chave ascending é opcional. Será a ordem de classificação padrão se nenhuma ordem for especificada.
Para obter mais informações, consulte Cláusula orderby.
Cláusula join
Use a cláusula join para associar e/ou combinar elementos de uma fonte de dados com elementos de outra
fonte de dados com base em uma comparação de igualdade entre as chaves especificadas em cada elemento. Na
LINQ, as operações join são executadas em sequências de objetos cujos elementos são de tipos diferentes. Após
ter unido duas sequências, você deve usar uma instrução select ou group para especificar qual elemento
armazenar na sequência de saída. Você também pode usar um tipo anônimo para combinar propriedades de cada
conjunto de elementos associados em um novo tipo para a sequência de saída. O exemplo a seguir associa
objetos prod cuja propriedade Category corresponde a uma das categorias na matriz de cadeias de caracteres
categories . Produtos cuja Category não corresponde a nenhuma cadeia de caracteres em categories são
filtrados. A instrução select projeta um novo tipo cujas propriedades são tiradas de cat e prod .
var categoryQuery =
from cat in categories
join prod in products on cat equals prod.Category
select new { Category = cat, Name = prod.Name };
Você também pode executar uma junção de grupo armazenando os resultados da operação join em uma
variável temporária usando a palavra-chave into. Para obter mais informações, consulte Cláusula join.
Cláusula let
Use a cláusula let para armazenar o resultado de uma expressão, como uma chamada de método, em uma nova
variável de intervalo. No exemplo a seguir, a variável de intervalo firstName armazena o primeiro elemento da
matriz de cadeias de caracteres que é retornado pelo Split .
string[] names = { "Svetlana Omelchenko", "Claire O'Donnell", "Sven Mortensen", "Cesar Garcia" };
IEnumerable<string> queryFirstNames =
from name in names
let firstName = name.Split(' ')[0]
select firstName;
Para obter mais informações, consulte Como executar uma subconsulta em uma operação de agrupamento.
Consulte também
Guia de programação em C#
LINQ (Consulta Integrada à Linguagem)
Palavras-chave de consulta (LINQ )
Visão geral de operadores de consulta padrão
LINQ em C#
04/11/2019 • 2 minutes to read • Edit Online
Esta seção contém links para tópicos que fornecem informações mais detalhadas sobre a LINQ.
Nesta seção
Introdução a consultas LINQ
Descreve as três partes da operação de consulta LINQ básica que são comuns a todas as linguagens e fontes de
dados.
LINQ e tipos genéricos
Fornece uma breve introdução sobre como os tipos genéricos são usados na LINQ.
Transformações de dados com LINQ
Descreve as várias maneiras que você pode transformar os dados recuperados em consultas.
Relacionamentos de tipo em operações de consulta LINQ
Descreve como tipos são preservados e/ou transformados nas três partes de uma operação de consulta LINQ
Sintaxe de consulta e sintaxe de método em LINQ
Compara a sintaxe de método e a sintaxe de consulta como dois modos para expressar uma consulta LINQ.
Recursos do C# que dão suporte a LINQ
Descreve os constructos da linguagem em C# que dão suporte ao LINQ.
Seções relacionadas
Expressões de consulta LINQ
Inclui uma visão geral de consultas na LINQ e fornece links para recursos adicionais.
Visão geral de operadores de consulta padrão
Apresenta os métodos padrão usados na LINQ.
Escrever consultas LINQ em C#
23/10/2019 • 7 minutes to read • Edit Online
Este artigo mostra as três maneiras de escrever uma consulta LINQ em C#:
1. Usar a sintaxe de consulta.
2. Usar a sintaxe do método.
3. Usar uma combinação da sintaxe de consulta e da sintaxe de método.
Os exemplos a seguir demonstram algumas consultas LINQ simples usando cada abordagem listada
anteriormente. Em geral, a regra é usar (1) sempre que possível e usar (2) e (3) sempre que necessário.
NOTE
Essas consultas funcionam em coleções na memória simples, no entanto, a sintaxe básica é idêntica àquela usada no LINQ to
Entities e no LINQ to XML.
// Query #1.
List<int> numbers = new List<int>() { 5, 4, 1, 3, 9, 8, 6, 7, 2, 0 };
// Query #2.
IEnumerable<int> orderingQuery =
from num in numbers
where num < 3 || num > 7
orderby num ascending
select num;
// Query #3.
string[] groupingQuery = { "carrots", "cabbage", "broccoli", "beans", "barley" };
IEnumerable<IGrouping<char, string>> queryFoodGroups =
from item in groupingQuery
group item by item[0];
Observe que o tipo das consultas é IEnumerable<T>. Todas essas consultas poderiam ser escritas usando var
conforme mostrado no exemplo a seguir:
var query = from num in numbers...
Em cada exemplo anterior, as consultas não são de fato executadas até você iterar na variável de consulta em uma
instrução foreach ou outra instrução. Para obter mais informações, consulte Introdução a Consultas LINQ.
// Query #5.
IEnumerable<int> concatenationQuery = numbers1.Concat(numbers2);
Se o método tiver os parâmetros Action ou Func, eles serão fornecidos na forma de uma expressão lambda, como
mostra o exemplo a seguir:
// Query #6.
IEnumerable<int> largeNumbersQuery = numbers2.Where(c => c > 15);
Nas consultas anteriores, apenas a Query #4 é executada imediatamente. Isso ocorre porque ele retorna um único
valor e não uma coleção IEnumerable<T> genérica. O próprio método tem que usar foreach para calcular seu
valor.
Cada uma das consultas anteriores pode ser escrita usando a tipagem implícita com var, como mostrado no
exemplo a seguir:
Como a Query #7 retorna um único valor e não uma coleção, a consulta é executada imediatamente.
A consulta anterior pode ser escrita usando a tipagem implícita com var , da seguinte maneira:
Consulte também
Passo a passo: Escrevendo consultas em C#
LINQ (Consulta Integrada à Linguagem)
Cláusula where
Consultar uma coleção de objetos
23/10/2019 • 3 minutes to read • Edit Online
Este exemplo mostra como executar uma consulta simples em uma lista de objetos Student . Cada objeto
Student contém algumas informações básicas sobre o aluno e uma lista que representa as pontuações do aluno
em quatro provas.
Este aplicativo serve como a estrutura para muitos outros exemplos nesta seção que usam as mesmas fontes de
dados students .
Exemplo
A consulta a seguir retorna os alunos que receberam uma pontuação de 90 ou mais em sua primeira prova.
Essa consulta é intencionalmente simples para que você possa testar. Por exemplo, você pode testar mais
condições na cláusula where ou usar uma clausula orderby para classificar os resultados.
Consulte também
LINQ (Consulta Integrada à Linguagem)
Interpolação de cadeia de caracteres
Como retornar uma consulta de um método (C# guia
de programação)
25/11/2019 • 3 minutes to read • Edit Online
Este exemplo mostra como retornar uma consulta de um método como o valor retornado e como um parâmetro
out .
Os objetos de consulta são combináveis, o que significa que você pode retornar uma consulta de um método. Os
objetos que representam consultas não armazenam a coleção resultante, mas as etapas para gerar os resultados
quando necessário. A vantagem de retornar objetos de consulta de métodos é que eles podem ser ainda mais
modificados e combinados. Portanto, qualquer valor retornado ou parâmetro out de um método que retorna uma
consulta também deve ter o tipo. Se um método materializa uma consulta em um tipo List<T> ou Array concreto,
considera-se que ele está retornando os resultados da consulta em vez da consulta em si. Uma variável de consulta
retornada de um método ainda pode ser combinada ou modificada.
Exemplo
No exemplo a seguir, o primeiro método retorna uma consulta como um valor retornado e o segundo método
retorna uma consulta como um parâmetro out . Observe que em ambos os casos é uma consulta que é retornada,
não os resultados da consulta.
class MQ
{
// QueryMethhod1 returns a query as its value.
IEnumerable<string> QueryMethod1(ref int[] ints)
{
var intsToStrings = from i in ints
where i > 4
select i.ToString();
return intsToStrings;
}
int[] nums = { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9 };
IEnumerable<string> myQuery2;
// QueryMethod2 returns a query as the value of its out parameter.
app.QueryMethod2(ref nums, out myQuery2);
Consulte também
LINQ (Consulta Integrada à Linguagem)
Armazenar os resultados de uma consulta na
memória
23/10/2019 • 2 minutes to read • Edit Online
Uma consulta é basicamente um conjunto de instruções sobre como recuperar e organizar os dados. As consultas
são executadas lentamente, conforme cada item subsequente no resultado é solicitado. Quando você usa foreach
para iterar os resultados, os itens são retornados conforme acessado. Para avaliar uma consulta e armazenar os
resultados sem executar um loop foreach , basta chamar um dos métodos a seguir na variável de consulta:
ToList
ToArray
ToDictionary
ToLookup
Recomendamos que ao armazenar os resultados da consulta, você atribua o objeto da coleção retornado a uma
nova variável conforme mostrado no exemplo a seguir:
Exemplo
class StoreQueryResults
{
static List<int> numbers = new List<int>() { 1, 2, 4, 6, 8, 10, 12, 14, 16, 18, 20 };
static void Main()
{
IEnumerable<int> queryFactorsOfFour =
from num in numbers
where num % 4 == 0
select num;
Consulte também
LINQ (Consulta Integrada à Linguagem)
Agrupar resultados de consultas
23/10/2019 • 10 minutes to read • Edit Online
O agrupamento é um dos recursos mais poderosos do LINQ. Os exemplos a seguir mostram como agrupar dados
de várias maneiras:
Por uma única propriedade.
Pela primeira letra de uma propriedade de cadeia de caracteres.
Por um intervalo numérico calculado.
Por predicado booliano ou outra expressão.
Por uma chave composta.
Além disso, as duas últimas consultas projetam seus resultados em um novo tipo anônimo que contém somente o
primeiro nome e sobrenome do aluno. Para obter mais informações, consulte a cláusula group.
Exemplo
Todos os exemplos neste tópico usam as seguintes fontes de dados e classes auxiliares.
Exemplo
O exemplo a seguir mostra como agrupar elementos de origem usando uma única propriedade do elemento como
a chave de grupo. Nesse caso, a chave é uma string , o sobrenome do aluno. Também é possível usar uma
subcadeia para a chave. A operação de agrupamento usa o comparador de igualdade padrão para o tipo.
Cole o seguinte método na classe StudentClass . Altere a instrução de chamada no método Main para
sc.GroupBySingleProperty() .
public void GroupBySingleProperty()
{
Console.WriteLine("Group by a single property in an object:");
Exemplo
O exemplo a seguir mostra como agrupar elementos de origem usando algo diferente de uma propriedade do
objeto para a chave de grupo. Neste exemplo, a chave é a primeira letra do sobrenome do aluno.
Cole o seguinte método na classe StudentClass . Altere a instrução de chamada no método Main para
sc.GroupBySubstring() .
public void GroupBySubstring()
{
Console.WriteLine("\r\nGroup by something other than a property of the object:");
var queryFirstLetters =
from student in students
group student by student.LastName[0];
Exemplo
O exemplo a seguir mostra como agrupar elementos de origem usando um intervalo numérico como a chave de
grupo. Em seguida, a consulta, projeta os resultados em um tipo anônimo que contém apenas o nome e o
sobrenome e o intervalo de percentil ao qual o aluno pertence. Um tipo anônimo é usado porque não é necessário
usar o objeto Student completo para exibir os resultados. GetPercentile é uma função auxiliar que calcula um
percentil com base na pontuação média do aluno. O método retorna um número inteiro entre 0 e 10.
Cole o seguinte método na classe StudentClass . Altere a instrução de chamada no método Main para
sc.GroupByRange() .
public void GroupByRange()
{
Console.WriteLine("\r\nGroup by numeric range and project into a new anonymous type:");
var queryNumericRange =
from student in students
let percentile = GetPercentile(student)
group new { student.FirstName, student.LastName } by percentile into percentGroup
orderby percentGroup.Key
select percentGroup;
Exemplo
O exemplo a seguir mostra como agrupar elementos de origem usando uma expressão de comparação booliana.
Neste exemplo, a expressão booliana testa se a pontuação média de provas do aluno é maior que 75. Como nos
exemplos anteriores, os resultados são projetados em um tipo anônimo porque o elemento de origem completo
não é necessário. Observe que as propriedades no tipo anônimo se tornam propriedades no membro Key e
podem ser acessadas pelo nome quando a consulta é executada.
Cole o seguinte método na classe StudentClass . Altere a instrução de chamada no método Main para
sc.GroupByBoolean() .
public void GroupByBoolean()
{
Console.WriteLine("\r\nGroup by a Boolean into two groups with string keys");
Console.WriteLine("\"True\" and \"False\" and project into a new anonymous type:");
var queryGroupByAverages = from student in students
group new { student.FirstName, student.LastName }
by student.ExamScores.Average() > 75 into studentGroup
select studentGroup;
Exemplo
O exemplo a seguir mostra como usar um tipo anônimo para encapsular uma chave que contém vários valores.
Neste exemplo, o primeiro valor da chave é a primeira letra do sobrenome do aluno. O segundo valor da chave é
um booliano que especifica se o aluno tirou mais que 85 na primeira prova. Você pode ordenar os grupos por
qualquer propriedade na chave.
Cole o seguinte método na classe StudentClass . Altere a instrução de chamada no método Main para
sc.GroupByCompositeKey() .
public void GroupByCompositeKey()
{
var queryHighScoreGroups =
from student in students
group student by new { FirstLetter = student.LastName[0],
Score = student.ExamScores[0] > 85 } into studentGroup
orderby studentGroup.Key.FirstLetter
select studentGroup;
/* Output:
Group and order by a compound key:
Name starts with A who scored more than 85
Terry Adams
Name starts with F who scored more than 85
Fadi Fakhouri
Hanying Feng
Name starts with G who scored more than 85
Cesar Garcia
Hugo Garcia
Name starts with G who scored less than 85
Debra Garcia
Name starts with M who scored more than 85
Sven Mortensen
Name starts with O who scored less than 85
Claire O'Donnell
Name starts with O who scored more than 85
Svetlana Omelchenko
Name starts with T who scored less than 85
Lance Tucker
Name starts with T who scored more than 85
Michael Tucker
Name starts with Z who scored more than 85
Eugene Zabokritski
*/
Consulte também
GroupBy
IGrouping<TKey,TElement>
LINQ (Consulta Integrada à Linguagem)
Cláusula group
Tipos Anônimos
Executar uma subconsulta em uma operação de agrupamento
Criar um grupo aninhado
Agrupando Dados
Criar um grupo aninhado
23/10/2019 • 2 minutes to read • Edit Online
O exemplo a seguir mostra como criar grupos aninhados em uma expressão de consulta LINQ. Cada grupo que é
criado de acordo com o ano do aluno ou nível de ensino, é subdividido em grupos com base nos nomes das
pessoas.
Exemplo
NOTE
Este exemplo contém referências a objetos que são definidos no código de exemplo em Consultar uma coleção de objetos.
public void QueryNestedGroups()
{
var queryNestedGroups =
from student in students
group student by student.Year into newGroup1
from newGroup2 in
(from student in newGroup1
group student by student.LastName)
group newGroup2 by newGroup1.Key;
Observe que três loops foreach aninhados são necessários para iterar sobre os elementos internos de um grupo
aninhado.
Consulte também
LINQ (Consulta Integrada à Linguagem)
Executar uma subconsulta em uma operação de
agrupamento
23/10/2019 • 2 minutes to read • Edit Online
Este artigo mostra duas maneiras diferentes de criar uma consulta que ordena os dados de origem em grupos e,
em seguida, realiza uma subconsulta em cada grupo individualmente. A técnica básica em cada exemplo é
agrupar os elementos de origem usando uma continuação chamada newGroup e, em seguida, gerar uma nova
subconsulta de newGroup . Essa subconsulta é executada em cada novo grupo criado pela consulta externa.
Observe que, nesse exemplo específico, a saída final não é um grupo, mas uma sequência simples de tipos
anônimos.
Para obter mais informações sobre como agrupar, consulte Cláusula group.
Para obter mais informações sobre continuações, consulte into. O exemplo a seguir usa uma estrutura de dados
na memória como a fonte de dados, mas os mesmos princípios se aplicam a qualquer tipo de fonte de dados do
LINQ.
Exemplo
NOTE
Este exemplo contém referências a objetos que são definidos no código de exemplo em Consultar uma coleção de objetos.
A consulta no snippet acima também pode ser escrita usando a sintaxe de método. O snippet de código a seguir
tem uma consulta semanticamente equivalente escrita usando a sintaxe de método.
public void QueryMaxUsingMethodSyntax()
{
var queryGroupMax = students
.GroupBy(student => student.Year)
.Select(studentGroup => new
{
Level = studentGroup.Key,
HighestScore = studentGroup.Select(student2 => student2.ExamScores.Average()).Max()
});
Consulte também
LINQ (Consulta Integrada à Linguagem)
Agrupar resultados por chaves contíguas
23/10/2019 • 8 minutes to read • Edit Online
O exemplo a seguir mostra como agrupar elementos em partes que representam subsequências de chaves
contíguas. Por exemplo, suponha que você receba a seguinte sequência de pares chave-valor:
CHAVE VALOR
Um We
Um think
Um that
B Linq
C is
Um really
B cool
B !
Exemplo
O exemplo a seguir mostra o método de extensão e o código do cliente que o usa:
using System;
using System.Collections.Generic;
using System.Linq;
namespace ChunkIt
{
// Static class to contain the extension methods.
public static class MyExtensions
{
public static IEnumerable<IGrouping<TKey, TSource>> ChunkBy<TSource, TKey>(this IEnumerable<TSource>
source, Func<TSource, TKey> keySelector)
{
return source.ChunkBy(keySelector, EqualityComparer<TKey>.Default);
}
// Make a new Chunk (group) object that initially has one GroupItem, which is a copy of the
current source element.
current = new Chunk<TKey, TSource>(key, enumerator, value => comparer.Equals(key,
keySelector(value)));
// Return the Chunk. A Chunk is an IGrouping<TKey,TSource>, which is the return value of the
ChunkBy method.
// At this point the Chunk only has the first element in its source sequence. The remaining
elements will be
// returned only when the client code foreach's over this chunk. See Chunk.GetEnumerator for
more info.
yield return current;
// Check to see whether (a) the chunk has made a copy of all its source elements or
// (b) the iterator has reached the end of the source sequence. If the caller uses an inner
// foreach loop to iterate the chunk items, and that loop ran to completion,
// then the Chunk.GetEnumerator method will already have made
// copies of all chunk items before we get here. If the Chunk.GetEnumerator loop did not
// enumerate all elements in the chunk, we need to do it here to avoid corrupting the iterator
// for clients that may be calling us on a separate thread.
if (current.CopyAllChunkElements() == noMoreSourceElements)
{
yield break;
}
}
}
// A Chunk is a contiguous group of one or more source elements that have the same key. A Chunk
// has a key and a list of ChunkItem objects, which are copies of the elements in the source sequence.
class Chunk<TKey, TSource> : IGrouping<TKey, TSource>
{
// INVARIANT: DoneCopyingChunk == true ||
// (predicate != null && predicate(enumerator.Current) && current.Value == enumerator.Current)
// A Chunk has a linked list of ChunkItems, which represent the elements in the current chunk. Each
ChunkItem
// has a reference to the next ChunkItem in the list.
class ChunkItem
{
public ChunkItem(TSource value)
{
Value = value;
}
public readonly TSource Value;
public ChunkItem Next = null;
}
// Flag to indicate the source iterator has reached the end of the source sequence.
internal bool isLastSourceElement = false;
// The end and beginning are the same until the list contains > 1 elements.
tail = head;
// Indicates that all chunk elements have been copied to the list of ChunkItems,
// and the source enumerator is either at the end, or else on an element with a new key.
// the tail of the linked list is set to null in the CopyNextChunkElement method if the
// key of the next element does not match the current chunk's key, or there are no more elements in
the source.
private bool DoneCopyingChunk => tail == null;
// If we are (a) at the end of the source, or (b) at the end of the current chunk
// then null out the enumerator and predicate for reuse with the next chunk.
// then null out the enumerator and predicate for reuse with the next chunk.
if (isLastSourceElement || !predicate(enumerator.Current))
{
enumerator = null;
predicate = null;
}
else
{
tail.Next = new ChunkItem(enumerator.Current);
}
// Called after the end of the last chunk was reached. It first checks whether
// there are more elements in the source sequence. If there are, it
// Returns true if enumerator for this chunk was exhausted.
internal bool CopyAllChunkElements()
{
while (true)
{
lock (m_Lock)
{
if (DoneCopyingChunk)
{
// If isLastSourceElement is false,
// it signals to the outer iterator
// to continue iterating.
return isLastSourceElement;
}
else
{
CopyNextChunkElement();
}
}
}
}
// Invoked by the inner foreach loop. This method stays just one step ahead
// of the client requests. It adds the next element of the chunk only after
// the clients requests the last element in the list so far.
public IEnumerator<TSource> GetEnumerator()
{
//Specify the initial element to enumerate.
ChunkItem current = head;
// A simple named type is used for easier viewing in the debugger. Anonymous types
// work just as well with the ChunkBy operator.
public class KeyValPair
{
public string Key { get; set; }
public string Value { get; set; }
}
class Program
{
// The source sequence.
public static IEnumerable<KeyValPair> list;
Para usar o método de extensão em seu projeto, copie a classe estática MyExtensions para um arquivo de código-
fonte novo ou existente e se for necessário, adicione uma diretiva using para o namespace em que ele está
localizado.
Consulte também
LINQ (Consulta Integrada à Linguagem)
Especificar filtros predicados dinamicamente em
tempo de execução
23/10/2019 • 4 minutes to read • Edit Online
Em alguns casos, você não sabe até o tempo de execução quantos predicados precisa aplicar aos elementos de
origem na cláusula where . Uma maneira de especificar dinamicamente vários filtros de predicados é usar o
método Contains, conforme mostrado no exemplo a seguir. O exemplo é construído de duas maneiras. Primeiro, o
projeto é executado filtrando valores que são fornecidos no programa. Em seguida, o projeto é executado
novamente usando a entrada fornecida em tempo de execução.
5. Adicione a seguinte linha ao método Main na classe DynamicPredicates , sob a declaração de ids .
QueryById(ids);
6. Execute o projeto.
7. A saída a seguir é exibida em uma janela do console:
Garcia: 114
O'Donnell: 112
Omelchenko: 111
8. A próxima etapa é executar o projeto novamente, desta vez usando a entrada inserida em tempo de
execução em vez da matriz ids . Altere QueryByID(ids) para QueryByID(args) no método Main .
9. Execute o projeto com os argumentos de linha de comando 122 117 120 115 . Quando o projeto é
executado, esses valores se tornam elementos de args , o parâmetro do método Main .
10. A saída a seguir é exibida em uma janela do console:
Adams: 120
Feng: 117
Garcia: 115
Tucker: 122
default:
break;
}
Console.WriteLine($"The following students are at level {year}");
foreach (Student name in studentQuery)
{
Console.WriteLine($"{name.LastName}: {name.ID}");
}
}
3. No método Main , substitua a chamada para QueryByID pela chamada a seguir, que envia o primeiro
elemento da matriz args como seu argumento: QueryByYear(args[0]) .
4. Execute o projeto com um argumento de linha de comando de um valor inteiro entre 1 e 4.
Consulte também
LINQ (Consulta Integrada à Linguagem)
Cláusula where
Executar junções internas
23/10/2019 • 14 minutes to read • Edit Online
Em termos de banco de dados relacionais, uma junção interna produz um conjunto de resultados no qual cada
elemento da primeira coleção aparece uma vez para todo elemento correspondente na segunda coleção. Se um
elemento na primeira coleção não tiver nenhum elemento correspondente, ele não aparece no conjunto de
resultados. O método Join, que é chamado pela cláusula join no C#, implementa uma junção interna.
Este artigo mostra como executar quatro variações de uma junção interna:
Uma junção interna simples que correlaciona os elementos de duas fontes de dados com base em uma
chave simples.
Uma junção interna simples que correlaciona os elementos de duas fontes de dados com base em uma
chave composta. Uma chave composta, que é uma chave que consiste em mais de um valor, permite que
você correlacione os elementos com base em mais de uma propriedade.
Uma junção múltipla na qual operações join sucessivas são acrescentadas umas às outras.
Uma junção interna que é implementada por meio de uma junção de grupo.
class Pet
{
public string Name { get; set; }
public Person Owner { get; set; }
}
/// <summary>
/// Simple inner join.
/// </summary>
public static void InnerJoinExample()
{
Person magnus = new Person { FirstName = "Magnus", LastName = "Hedlund" };
Person terry = new Person { FirstName = "Terry", LastName = "Adams" };
Person charlotte = new Person { FirstName = "Charlotte", LastName = "Weiss" };
Person arlene = new Person { FirstName = "Arlene", LastName = "Huff" };
Person rui = new Person { FirstName = "Rui", LastName = "Raposo" };
Observe que o objeto Person cujo LastName é "Huff" não aparecerá no conjunto de resultados porque não há
nenhum objeto Pet que tenha Pet.Owner igual a esse Person .
class Employee
{
public string FirstName { get; set; }
public string LastName { get; set; }
public int EmployeeID { get; set; }
}
class Student
{
public string FirstName { get; set; }
public string LastName { get; set; }
public int StudentID { get; set; }
}
/// <summary>
/// Performs a join operation using a composite key.
/// </summary>
public static void CompositeKeyJoinExample()
{
// Create a list of employees.
List<Employee> employees = new List<Employee> {
new Employee { FirstName = "Terry", LastName = "Adams", EmployeeID = 522459 },
new Employee { FirstName = "Charlotte", LastName = "Weiss", EmployeeID = 204467 },
new Employee { FirstName = "Magnus", LastName = "Hedland", EmployeeID = 866200 },
new Employee { FirstName = "Vernette", LastName = "Price", EmployeeID = 437139 } };
// Join the two data sources based on a composite key consisting of first and last name,
// to determine which employees are also students.
IEnumerable<string> query = from employee in employees
join student in students
on new { employee.FirstName, employee.LastName }
equals new { student.FirstName, student.LastName }
select employee.FirstName + " " + employee.LastName;
A segunda cláusula join em C# correlaciona os tipos anônimos retornados pela primeira junção com objetos
Dog na lista de cães fornecida, com base em uma chave composta que consiste na propriedade Owner do tipo
Person e na primeira letra do nome do animal. Ela retorna uma sequência de tipos anônimos que contêm as
propriedades Cat.Name e Dog.Name de cada par correspondente. Como esta é uma junção interna, apenas os
objetos da primeira fonte de dados que têm uma correspondência na segunda fonte de dados são retornados.
class Person
{
public string FirstName { get; set; }
public string LastName { get; set; }
}
class Pet
{
public string Name { get; set; }
public Person Owner { get; set; }
}
Dog fourwheeldrive = new Dog { Name = "Four Wheel Drive", Owner = phyllis };
Dog duke = new Dog { Name = "Duke", Owner = magnus };
Dog denim = new Dog { Name = "Denim", Owner = terry };
Dog wiley = new Dog { Name = "Wiley", Owner = charlotte };
Dog snoopy = new Dog { Name = "Snoopy", Owner = rui };
Dog snickers = new Dog { Name = "Snickers", Owner = arlene };
// The first join matches Person and Cat.Owner from the list of people and
// cats, based on a common Person. The second join matches dogs whose names start
// with the same letter as the cats that have the same owner.
var query = from person in people
join cat in cats on person equals cat.Owner
join dog in dogs on
new { Owner = person, Letter = cat.Name.Substring(0, 1) }
equals new { dog.Owner, Letter = dog.Name.Substring(0, 1) }
select new { CatName = cat.Name, DogName = dog.Name };
class Person
{
public string FirstName { get; set; }
public string LastName { get; set; }
}
class Pet
{
public string Name { get; set; }
public Person Owner { get; set; }
}
/// <summary>
/// Performs an inner join by using GroupJoin().
/// </summary>
public static void InnerGroupJoinExample()
{
Person magnus = new Person { FirstName = "Magnus", LastName = "Hedlund" };
Person terry = new Person { FirstName = "Terry", LastName = "Adams" };
Person charlotte = new Person { FirstName = "Charlotte", LastName = "Weiss" };
Person arlene = new Person { FirstName = "Arlene", LastName = "Huff" };
Consulte também
Join
GroupJoin
Executar junções agrupadas
Executar junções externas esquerdas
Tipos anônimos
Executar junções agrupadas
23/10/2019 • 6 minutes to read • Edit Online
A junção de grupo é útil para a produção de estruturas de dados hierárquicos. Ela combina cada elemento da
primeira coleção com um conjunto de elementos correlacionados da segunda coleção.
Por exemplo, uma classe ou uma tabela de banco de dados relacional chamada Student pode conter dois campos:
Id e Name . Uma segunda classe ou tabela de banco de dados relacional chamada Course pode conter dois
campos: StudentId e CourseTitle . Uma junção de grupo dessas duas fontes de dados, com base na
correspondência de Student.Id e Course.StudentId , agruparia cada Student com uma coleção de objetos
Course (que pode estar vazia).
NOTE
Cada elemento da primeira coleção aparece no conjunto de resultados de uma junção de grupo, independentemente de se
os elementos correlacionados encontram-se na segunda coleção. Caso nenhum elemento correlacionado seja encontrado, a
sequência de elementos correlacionados desse elemento ficará vazia. O seletor de resultado, portanto, tem acesso a todos
os elementos da primeira coleção. Isso difere do seletor de resultado de uma junção que não é de grupo, que não pode
acessar os elementos da primeira coleção que não têm correspondência na segunda coleção.
O primeiro exemplo neste artigo mostra como executar uma junção de grupo. O segundo exemplo mostra como
usar uma junção de grupo para criar elementos XML.
class Pet
{
public string Name { get; set; }
public Person Owner { get; set; }
}
/// <summary>
/// This example performs a grouped join.
/// </summary>
public static void GroupJoinExample()
{
Person magnus = new Person { FirstName = "Magnus", LastName = "Hedlund" };
Person terry = new Person { FirstName = "Terry", LastName = "Adams" };
Person charlotte = new Person { FirstName = "Charlotte", LastName = "Weiss" };
Person arlene = new Person { FirstName = "Arlene", LastName = "Huff" };
class Person
{
public string FirstName { get; set; }
public string LastName { get; set; }
}
class Pet
{
public string Name { get; set; }
public Person Owner { get; set; }
}
/// <summary>
/// This example creates XML output from a grouped join.
/// </summary>
public static void GroupJoinXMLExample()
{
Person magnus = new Person { FirstName = "Magnus", LastName = "Hedlund" };
Person terry = new Person { FirstName = "Terry", LastName = "Adams" };
Person charlotte = new Person { FirstName = "Charlotte", LastName = "Weiss" };
Person arlene = new Person { FirstName = "Arlene", LastName = "Huff" };
// Create XML to display the hierarchical organization of people and their pets.
XElement ownersAndPets = new XElement("PetOwners",
from person in people
join pet in pets on person equals pet.Owner into gj
select new XElement("Person",
new XAttribute("FirstName", person.FirstName),
new XAttribute("LastName", person.LastName),
from subpet in gj
select new XElement("Pet", subpet.Name)));
Console.WriteLine(ownersAndPets);
}
Uma junção externa esquerda é uma junção em que cada elemento da primeira coleção é retornado, mesmo que
ele tenha elementos correlacionados na segunda coleção. É possível usar o LINQ para executar uma junção
externa esquerda chamando o método DefaultIfEmpty nos resultados de uma junção de grupo.
Exemplo
O exemplo a seguir demonstra como usar o método DefaultIfEmpty nos resultados de uma junção de grupo para
executar uma junção externa esquerda.
A primeira etapa da produção de uma junção externa esquerda de duas coleções é executar uma junção interna
usando uma junção de grupo. (Consulte Executar junções internas para obter uma explicação desse processo.)
Neste exemplo, a lista de objetos Person é associada internamente à lista de objetos Pet com base em um
objeto Person correspondente a Pet.Owner .
A segunda etapa é incluir cada elemento da primeira coleção (esquerda) no conjunto de resultados, mesmo que
esse elemento não tenha nenhuma correspondência na coleção direita. Isso é feito chamando DefaultIfEmpty em
cada sequência de elementos correspondentes da junção de grupo. Neste exemplo, DefaultIfEmpty é chamado
em cada sequência de objetos Pet correspondentes. O método retorna uma coleção que contém um valor
padrão único se a sequência de objetos Pet correspondentes estiver vazia para qualquer objeto Person ,
garantindo assim que cada objeto Person seja representado no conjunto de resultados.
NOTE
O valor padrão para um tipo de referência é null ; portanto, o exemplo procura uma referência nula antes de acessar cada
elemento de cada coleção Pet .
class Person
{
public string FirstName { get; set; }
public string LastName { get; set; }
}
class Pet
{
public string Name { get; set; }
public Person Owner { get; set; }
}
Consulte também
Join
GroupJoin
Executar junções internas
Executar junções agrupadas
Tipos anônimos
Ordenar os resultados de uma cláusula join
23/10/2019 • 2 minutes to read • Edit Online
Este exemplo mostra como ordenar os resultados de uma operação de junção. Observe que a ordenação é
executada após a junção. Embora você possa usar uma cláusula orderby com uma ou mais sequências de origem
antes da junção, normalmente não é recomendável. Alguns provedores LINQ não podem preservar essa ordem
após a junção.
Exemplo
Esta consulta cria uma junção de grupos e classifica os grupos com base no elemento de categoria, que ainda está
no escopo. Dentro do inicializador de tipo anônimo, uma subconsulta ordena todos os elementos de
correspondência da sequência de produtos.
class HowToOrderJoins
{
#region Data
class Product
{
public string Name { get; set; }
public int CategoryID { get; set; }
}
class Category
{
public string Name { get; set; }
public int ID { get; set; }
}
void OrderJoin1()
{
var groupJoinQuery2 =
from category in categories
join prod in products on category.ID equals prod.CategoryID into prodGroup
orderby category.Name
select new
{
Category = category.Name,
Products = from prod2 in prodGroup
orderby prod2.Name
select prod2
};
Consulte também
LINQ (Consulta Integrada à Linguagem)
Cláusula orderby
Cláusula join
Unir usando chaves compostas
23/10/2019 • 2 minutes to read • Edit Online
Este exemplo mostra como realizar operações de junção nas quais você deseja usar mais de uma chave para
definir uma correspondência. Isso é realizado por meio de uma chave composta. Uma chave composta é criada
como um tipo anônimo ou como um tipo nomeado com os valores que você deseja comparar. Se a variável de
consulta será passada entre limites de método, use um tipo nomeado que substitui Equals e GetHashCode para a
chave. Os nomes das propriedades e a ordem em que elas ocorrem, devem ser idênticas em cada chave.
Exemplo
O exemplo a seguir demonstra como usar uma chave composta para unir dados de três tabelas:
A inferência de tipos em chaves compostas depende dos nomes das propriedades nas chaves e da ordem em que
elas ocorrem. Quando as propriedades nas sequências de origem não têm os mesmos nomes, você precisa
atribuir novos nomes nas chaves. Por exemplo, se a tabela Orders e a tabela OrderDetails usaram nomes
diferentes para suas colunas, você poderia criar chaves compostas ao atribuir nomes idênticos nos tipos anônimos:
Consulte também
LINQ (Consulta Integrada à Linguagem)
Cláusula join
Cláusula group
Executar operações de junção personalizadas
23/10/2019 • 7 minutes to read • Edit Online
Este exemplo mostra como executar operações de junção que não são possíveis com a cláusula join . Em uma
expressão de consulta, a cláusula join é limitada e otimizada para junções por igualdade, que são, de longe, o tipo
de operação de junção mais comum. Ao realizar uma junção por igualdade, provavelmente você terá o melhor
desempenho usando a cláusula join .
No entanto, a cláusula join não pode ser usada nos seguintes casos:
Quando a junção se baseia em uma expressão de desigualdade (uma junção que não é por igualdade).
Quando a junção se baseia em mais de uma expressão de igualdade ou desigualdade.
Quando for necessário introduzir uma variável de intervalo temporária para a sequência do lado direito
(interna) antes da operação de junção.
Para executar junções que não são junções por igualdade, você pode usar várias cláusulas from para introduzir
cada fonte de dados de forma independente. Em seguida, você aplica uma expressão de predicado em uma
cláusula where à variável de intervalo para cada fonte. A expressão também pode assumir a forma de uma
chamada de método.
NOTE
Não confunda esse tipo de operação de junção personalizada com o uso de várias cláusulas from para acessar coleções
internas. Para obter mais informações, consulte Cláusula join.
Exemplo
O primeiro método no exemplo a seguir mostra uma união cruzada simples. Uniões cruzadas devem ser usadas
com cuidado porque podem produzir conjuntos de resultados muito grandes. No entanto, elas podem ser úteis em
alguns cenários para criar sequências de origem em que são executadas consultas adicionais.
O segundo método produz uma sequência de todos os produtos cuja ID da categoria está na lista de categorias no
lado esquerdo. Observe o uso da cláusula let e do método Contains para criar uma matriz temporária. Também
é possível criar a matriz antes da consulta e eliminar a primeira cláusula from .
class CustomJoins
{
#region Data
class Product
{
public string Name { get; set; }
public int CategoryID { get; set; }
}
class Category
{
public string Name { get; set; }
public int ID { get; set; }
}
void CrossJoin()
{
var crossJoinQuery =
from c in categories
from p in products
select new { c.ID, p.Name };
void NonEquijoin()
{
var nonEquijoinQuery =
from p in products
let catIds = from c in categories
select c.ID
where catIds.Contains(p.CategoryID) == true
select new { Product = p.Name, CategoryID = p.CategoryID };
Console.WriteLine("Non-equijoin query:");
foreach (var v in nonEquijoinQuery)
{
Console.WriteLine($"{v.CategoryID,-5}{v.Product}");
}
}
}
/* Output:
Cross Join Query:
1 Tea
1 Mustard
1 Pickles
1 Carrots
1 Bok Choy
1 Bok Choy
1 Peaches
1 Melons
1 Ice Cream
1 Mackerel
2 Tea
2 Mustard
2 Pickles
2 Carrots
2 Bok Choy
2 Peaches
2 Melons
2 Ice Cream
2 Mackerel
3 Tea
3 Mustard
3 Pickles
3 Carrots
3 Bok Choy
3 Peaches
3 Melons
3 Ice Cream
3 Mackerel
Non-equijoin query:
1 Tea
2 Mustard
2 Pickles
3 Carrots
3 Bok Choy
Press any key to exit.
*/
Exemplo
No exemplo a seguir, a consulta deve unir duas sequências com base nas chaves correspondentes que, no caso da
sequência interna (lado direito), não podem ser obtidas antes da cláusula join. Se essa junção tiver sido executada
com uma cláusula join , o método Split precisará ser chamado para cada elemento. O uso de várias cláusulas
from permite que a consulta evite a sobrecarga da chamada de método repetida. No entanto, como join é
otimizado, neste caso em particular ainda pode ser mais rápido do que usar várias cláusulas from . Os resultados
variam dependendo principalmente do quanto a chamada de método é cara.
class MergeTwoCSVFiles
{
static void Main()
{
// See section Compiling the Code for information about the data files.
string[] names = System.IO.File.ReadAllLines(@"../../../names.csv");
string[] scores = System.IO.File.ReadAllLines(@"../../../scores.csv");
class Student
{
public string FirstName { get; set; }
public string LastName { get; set; }
public int ID { get; set; }
public List<int> ExamScores { get; set; }
}
/* Output:
The average score of Omelchenko Svetlana is 82.5.
The average score of O'Donnell Claire is 72.25.
The average score of Mortensen Sven is 84.5.
The average score of Garcia Cesar is 88.25.
The average score of Garcia Debra is 67.
The average score of Fakhouri Fadi is 92.25.
The average score of Feng Hanying is 88.
The average score of Garcia Hugo is 85.75.
The average score of Tucker Lance is 81.75.
The average score of Adams Terry is 85.25.
The average score of Zabokritski Eugene is 83.
The average score of Tucker Michael is 92.
*/
Consulte também
LINQ (Consulta Integrada à Linguagem)
Cláusula join
Ordenar os resultados de uma cláusula join
Manipular valores nulos em expressões de consulta
08/11/2019 • 2 minutes to read • Edit Online
Este exemplo mostra como tratar os possíveis valores nulos em coleções de origem. Uma coleção de objetos, tal
como uma IEnumerable<T>, pode conter elementos cujo valor é null. Se uma coleção de origem for nula ou
contiver um elemento cujo valor for null e sua consulta não lidar com valores null, uma NullReferenceException
será gerada ao executar a consulta.
Exemplo
Você pode escrever o código defensivamente para evitar uma exceção de referência nula conforme mostrado no
exemplo a seguir:
var query1 =
from c in categories
where c != null
join p in products on c.ID equals
p?.CategoryID
select new { Category = c.Name, Name = p.Name };
No exemplo anterior, a cláusula where filtra todos os elementos nulos na sequência de categorias. Essa técnica é
independente da verificação de nulos na cláusula join. A expressão condicional com null nesse exemplo funciona
porque Products.CategoryID é do tipo int? que é uma abreviação para Nullable<int> .
Exemplo
Em uma cláusula join, se apenas uma das chaves de comparação for um tipo de valor que permite valor nulo, você
pode converter a outra para um tipo que permite valor nulo na expressão de consulta. No exemplo a seguir,
suponha que EmployeeID é uma coluna que contém os valores do tipo int? :
Consulte também
Nullable<T>
LINQ (Consulta Integrada à Linguagem)
Tipos de valor anuláveis
Tratar exceções em expressões de consulta
23/10/2019 • 4 minutes to read • Edit Online
É possível chamar qualquer método no contexto de uma expressão de consulta. No entanto, é recomendável que
você evite chamar qualquer método em uma expressão de consulta que possa criar um efeito colateral, como
modificar o conteúdo da fonte de dados ou gerar uma exceção. Este exemplo mostra como evitar exceções ao
chamar métodos em uma expressão de consulta, sem violar as diretrizes gerais sobre tratamento de exceção do
.NET. Essas diretrizes declaram que é aceitável capturar uma exceção específica quando você entende por que ela é
gerada em um determinado contexto. Para obter mais informações, consulte Melhores práticas para exceções.
O último exemplo mostra como tratar os casos em que é necessário lançar uma exceção durante a execução de
uma consulta.
Exemplo
O exemplo a seguir mostra como mover o código de tratamento de exceção para fora de uma expressão de
consulta. Isso só é possível quando o método não depende de nenhuma variável que seja local para a consulta.
class ExceptionsOutsideQuery
{
static void Main()
{
// DO THIS with a datasource that might
// throw an exception. It is easier to deal with
// outside of the query expression.
IEnumerable<int> dataSource;
try
{
dataSource = GetData();
}
catch (InvalidOperationException)
{
// Handle (or don't handle) the exception
// in the way that is appropriate for your application.
Console.WriteLine("Invalid operation");
goto Exit;
}
class QueryThatThrows
{
static void Main()
{
// Data source.
string[] files = { "fileA.txt", "fileB.txt", "fileC.txt" };
Consulte também
LINQ (Consulta Integrada à Linguagem)
Programação assíncrona
30/10/2019 • 19 minutes to read • Edit Online
Se você tiver qualquer necessidade vinculada à E/S (como a solicitação de dados de uma rede ou o acesso a um
banco de dados), você desejará usar a programação assíncrona. Você também pode ter código vinculado à CPU,
como a execução de um cálculo dispendioso, que também é um bom cenário para escrever código assíncrono.
O C# tem um modelo de programação assíncrono em nível de linguagem que permite escrever facilmente o
código assíncrono sem precisar manipular retornos de chamada ou estar em conformidade com uma biblioteca
que dá suporte à assincronia. Ele segue o que é conhecido como TAP (Padrão assíncrono baseado em tarefa).
E pronto! O código expressa a intenção (baixar alguns dados de forma assíncrona) sem enroscar na interação
com objetos de tarefa.
Exemplo vinculado à CPU: executar um cálculo para um jogo
Digamos que você está escrevendo um jogo para dispositivo móvel em que, ao pressionar um botão, poderá
causar danos a muitos inimigos na tela. A realização do cálculo de dano pode ser dispendiosa e fazê-lo no thread
da interface do usuário faria com que o jogo parecesse pausar durante a realização do cálculo!
A melhor maneira de lidar com isso é iniciar um thread em segundo plano que faz o trabalho usando Task.Run e
await seu resultado. Isso permitirá que a interface do usuário pareça suave enquanto o trabalho está sendo feito.
E pronto. Esse código expressa claramente a intenção do evento de clique do botão. Ele não requer o
gerenciamento manual de um thread em segundo plano e ele faz isso sem bloqueios.
O que acontece nos bastidores
Há muitas partes se movendo nos locais em que as operações assíncronas acontecem. Se você estiver curioso
sobre o que está acontecendo nos bastidores de Task e Task<T> , dê uma olhada no artigo Programação
assíncrona em detalhes para obter mais informações.
No lado C# das coisas, o compilador transforma seu código em uma máquina de estado que mantém o controle
de coisas, como transferir a execução quando uma await é alcançada e continuar a execução quando um
trabalho em segundo plano for concluído.
Para os que gostam da teoria, essa é uma implementação do Modelo Promise de assincronia.
Mais exemplos
Os exemplos a seguir demonstram várias maneiras para escrever código assíncrono no C#. Elas abordam alguns
cenários diferentes que você pode encontrar.
Extrair dados de uma rede
Este snippet de código baixa o HTML da página inicial em www.dotnetfoundation.org e conta o número de vezes
que a cadeia de caracteres ".NET" ocorre no HTML. Ele usa o ASP.NET MVC para definir um método do
controlador da Web que realiza essa tarefa, retornando o número.
NOTE
Se você pretende fazer análise de HTML no código de produção, não use expressões regulares. Use uma biblioteca de
análise.
[HttpGet]
[Route("DotNetCount")]
public async Task<int> GetDotNetCountAsync()
{
// Suspends GetDotNetCountAsync() to allow the caller (the web server)
// to accept another request, rather than blocking on this one.
var html = await _httpClient.GetStringAsync("https://dotnetfoundation.org");
Aqui está o mesmo cenário escrito para um aplicativo universal do Windows, que executa a mesma tarefa
quando um botão for pressionado:
private readonly HttpClient _httpClient = new HttpClient();
// Any other work on the UI thread can be done here, such as enabling a Progress Bar.
// This is important to do here, before the "await" call, so that the user
// sees the progress bar before execution of this method is yielded.
NetworkProgressBar.IsEnabled = true;
NetworkProgressBar.Visibility = Visibility.Visible;
NetworkProgressBar.IsEnabled = false;
NetworkProgressBar.Visibility = Visibility.Collapsed;
}
Aqui está outro jeito de escrever isso, de forma um pouco mais sucinta, usando LINQ:
public async Task<User> GetUserAsync(int userId)
{
// Code omitted:
//
// Given a user Id {userId}, retrieves a User object corresponding
// to the entry in the database with {userId} as its Id.
}
Embora seja menos código, tome cuidado ao misturar LINQ com código assíncrono. Como o LINQ utiliza a
execução adiada (lenta), as chamadas assíncronas não acontecerão imediatamente como em um loop foreach() ,
a menos que você force a sequência gerada a iterar com uma chamada a .ToList() ou .ToArray() .
É importante ter isso em mente. Se await não for usado no corpo de um método async , o compilador do C#
gerará um aviso, mas o código será compilado e executado como se fosse um método normal. Observe que isso
também seria extremamente ineficiente, pois a máquina de estado, gerada pelo compilador do C# para o método
assíncrono, não realizaria nada.
Você deve adicionar "Async" como o sufixo de cada nome de método assíncrono que escrever.
Essa é a convenção usada no .NET para diferenciar mais facilmente os métodos síncronos e assíncronos.
Observe que isso não se aplica, necessariamente, a alguns métodos que não são explicitamente chamados pelo
seu código (como manipuladores de eventos ou métodos do controlador da Web). Como eles não são chamados
explicitamente pelo seu código, ser explícito em relação à sua nomenclatura não é tão importante.
async void O só deve ser usado para manipuladores de eventos.
O async void é a única maneira de permitir que os manipuladores de eventos assíncronos trabalhem, pois os
eventos não têm tipos de retorno (portanto, não podem fazer uso de Task e Task<T> ). Qualquer outro uso de
async void não segue o modelo TAP e pode ser um desafio utilizá-lo, como:
As exceções lançadas em um método async void não podem ser capturadas fora desse método.
Os métodos async void são muito difíceis de testar.
Os métodos async void poderão causar efeitos colaterais indesejados se o chamador não estiver
esperando que eles sejam assíncronos.
Vá com cuidado ao usar lambdas assíncronas em expressões LINQ
As expressões lambda em LINQ usam a execução adiada, o que significa que o código poderia acabar
executando em um momento que você não está esperando. A introdução de tarefas de bloqueio no meio disso
poderia facilmente resultar em um deadlock, se não estivessem escritas corretamente. Além disso, o
aninhamento de código assíncrono dessa maneira também pode dificultar a ponderação a respeito da execução
do código. A assíncrona e a LINQ são poderosas, mas devem ser usadas de uma maneira mais cuidadosa e clara
possível.
Escrever código que aguarda tarefas de uma maneira sem bloqueio
Bloquear o thread atual como um meio de aguardar a conclusão de uma tarefa pode resultar em deadlocks e
threads de contexto bloqueados e pode exigir tratamento de erros significativamente mais complexo. A tabela a
seguir fornece diretrizes de como lidar com a espera de tarefas de uma forma sem bloqueio:
Uma meta recomendada é alcançar a Transparência referencial completa ou quase completa em seu código. Isso
resultará em uma base de código extremamente previsível, testável e de fácil manutenção.
Outros recursos
A Programação assíncrona em detalhes fornece mais informações sobre o funcionamento de Tarefas.
Programação assíncrona com async e await (C#)
Seis dicas essenciais para a programação assíncrona de Lucian Wischik é um ótimo recurso para a
programação assíncrona
Correspondência padrão
30/10/2019 • 24 minutes to read • Edit Online
Os padrões testam um valor que tem uma determinada forma e podem extrair informações do valor quando ele
tem a forma correspondente. A correspondência de padrões fornece uma sintaxe mais concisa para algoritmos
que você já usa atualmente. Você já cria algoritmos de correspondência de padrões usando a sintaxe existente.
Você escreve instruções if ou switch que testam os valores. Então, quando essas instruções correspondem,
você extrai e usa informações desse valor. Os novos elementos de sintaxe são extensões para as instruções com
as quais você já está familiarizado: is e switch . Essas novas extensões combinam o teste de um valor e a
extração dessas informações.
Neste artigo, vamos examinar a nova sintaxe para mostrar a você como ela possibilita um código conciso e
legível. A correspondência de padrões habilita expressões em que os dados e o código são separados, diferente
dos designs orientados a objeto em que os dados e os métodos que os manipulam estão intimamente ligados.
Para ilustrar essas novas expressões, vamos trabalhar com estruturas que representam formas geométricas
usando instruções de correspondência de padrões. Provavelmente você está familiarizado com a criação de
hierarquias de classe com a criação de métodos virtuais e métodos substituídos para personalizar o
comportamento do objeto com base no tipo de tempo de execução do objeto.
Essas técnicas não são possíveis para dados não estruturados em uma hierarquia de classe. Quando os dados e
métodos são separados, você precisa de outras ferramentas. Os novos constructos de correspondência de
padrões permitem uma sintaxe mais clara para examinar dados e manipular o fluxo de controle com base em
qualquer condição desses dados. Você já escreve instruções if e switch que testam o valor da variável. Você
escreve instruções is que testam o tipo da variável. A correspondência de padrões adiciona novos recursos a
essas instruções.
Neste artigo, você criará um método que calcula a área de diferentes formas geométricas. Mas fará isso sem
recorrer às técnicas orientadas a objeto e sem criar uma hierarquia de classe para as diferentes formas. Em vez
disso, você usará correspondência de padrões. Conforme percorrer esse exemplo, contraste esse código com
como ele seria estruturado em uma hierarquia de objeto. Quando os dados que você deve consultar e manipular
não são uma hierarquia de classe, a correspondência de padrões permite designs elegantes.
Em vez de iniciar com uma definição de forma abstrata e adicionar classes de forma específicas diferentes, vamos
começar com os dados simples, apenas definições de dados para cada uma das formas geométricas:
public class Square
{
public double Side { get; }
Dessas estruturas, vamos escrever um método que calcula a área de uma forma.
O código acima é uma expressão clássica do padrão de tipo: você está testando uma variável para determinar seu
tipo e adotando uma ação diferente com base no tipo.
Esse código se torna mais simples usando extensões para a expressão is para atribuir uma variável se o teste
tiver êxito:
Nesta versão atualizadas, a expressão is testa a variável e a atribui a uma nova variável do tipo adequado. Além
disso, observe que essa versão inclui o tipo Rectangle , que é um struct . A nova expressão is funciona com
tipos de valor, bem como com tipos de referência.
As regras da linguagem para expressões de correspondência de padrões ajudam a evitar usar incorretamente os
resultados de uma expressão de correspondência. No exemplo acima, as variáveis s , c e r estão somente no
escopo e são atribuídas definitivamente quando as expressões de correspondência de padrão têm resultados
true . Se você tentar usar qualquer variável em outro local, seu código gerará erros de compilador.
Vamos examinar ambas as regras detalhadamente, começando com o escopo. A variável c está no escopo
somente no branch else da primeira instrução if . A variável s está no escopo no método
ComputeAreaModernIs . Isso ocorre porque cada branch de uma instrução if estabelece um escopo separado para
as variáveis. No entanto, a instrução if em si não. Isso significa que as variáveis declaradas na instrução if
estão no mesmo escopo que a instrução if (o método nesse caso.) Esse comportamento não é específico da
correspondência de padrões, mas é o comportamento definido para escopos de variáveis e instruções if e
else .
As variáveis c e s são atribuídas quando as respectivas instruções if são verdadeiras por causa do
mecanismo atribuído definitivamente quando elas são verdadeiras.
TIP
Os exemplos neste tópico usam o constructo recomendado quando uma expressão is de correspondência de padrão
atribui definitivamente a variável correspondente no branch true da instrução if . Você poderia reverter a lógica
dizendo que if (!(shape is Square s)) e a variável s seriam definitivamente atribuídas apenas no branch false .
Embora isso seja válido no C#, não é recomendável porque é mais confuso para acompanhar a lógica.
Essas regras significam que é improvável que você acesse acidentalmente o resultado de uma expressão de
correspondência de padrão quando esse padrão não foi atendido.
O único padrão com suporte pela instrução switch era o padrão de constante. Ele era ainda mais limitado a tipos
numéricos e ao tipo string . Essas restrições foram removidas e agora você pode escrever uma instrução
switch usando o padrão de tipo:
NOTE
As instruções goto para pular para outro rótulo são válidas somente para o padrão de constante, a instrução switch
clássica.
Há novas regras importantes regendo a instrução switch . As restrições no tipo da variável na expressão switch
foram removidas. Qualquer tipo, como object neste exemplo, pode ser usado. As expressões case não são mais
limitadas a valores de constantes. Remover essa limitação significa que reordenar seções switch pode alterar o
comportamento do programa.
Quando limitado a valores de constantes, no máximo um rótulo case poderia corresponder ao valor da
expressão switch . Combine isso com a regra de que cada seção switch não deve passar para a próxima seção e
isso resulta em que as seções switch poderiam ser reorganizadas em qualquer ordem sem afetar o
comportamento. Agora, com expressões switch mais generalizadas, a ordem de cada seção é importante. As
expressões switch são avaliadas na ordem textual. A execução transfere o primeiro rótulo switch que
corresponde à expressão switch .
O case default será executado somente se nenhum outro rótulo case corresponder. O case default é avaliado
por último, independentemente de sua ordem textual. Se não houver nenhum case default e nenhuma das
outras instruções case corresponder, a execução continuará na instrução após a instrução switch . Nenhum dos
códigos de rótulos case será executado.
case Square s:
return s.Side * s.Side;
case Circle c:
return c.Radius * c.Radius * Math.PI;
default:
throw new ArgumentException(
message: "shape is not a recognized shape",
paramName: nameof(shape));
}
}
Essa alteração demonstra alguns pontos importantes sobre a nova sintaxe. Primeiro, vários rótulos case podem
ser aplicados a uma seção switch . O bloco de instrução é executado quando qualquer um desses rótulos é true .
Neste exemplo, se a expressão switch é um círculo ou um quadrado com área 0, o método retorna a constante 0.
Este exemplo apresenta duas variáveis diferentes nos dois rótulos case para o primeiro bloco switch . Observe
que as instruções neste bloco switch não usam as variáveis c (para o círculo) ou s (para o quadrado).
Nenhuma dessas variáveis é atribuída definitivamente nesse bloco switch . Se algum desses casos corresponder,
claramente uma das variáveis foi atribuída. No entanto, é impossível dizer qual foi atribuída em tempo de
compilação, porque ambos os casos poderiam corresponder em tempo de execução. Por esse motivo, na maioria
das vezes que você usar vários rótulos case para o mesmo bloco, não introduzirá uma nova variável na
instrução case ou apenas usará a variável na cláusula when .
Depois de adicionar essas formas com a área 0, vamos adicionar mais alguns tipos de forma: um retângulo e um
triângulo:
case Square s:
return s.Side * s.Side;
case Circle c:
return c.Radius * c.Radius * Math.PI;
case Triangle t:
return t.Base * t.Height / 2;
case Rectangle r:
return r.Length * r.Height;
default:
throw new ArgumentException(
message: "shape is not a recognized shape",
paramName: nameof(shape));
}
}
Esse conjunto de alterações adiciona rótulos case para o caso de degeneração e rótulos e blocos para cada uma
das novas formas.
Por fim, você pode adicionar um case null para garantir que o argumento não seja null :
public static double ComputeArea_Version5(object shape)
{
switch (shape)
{
case Square s when s.Side == 0:
case Circle c when c.Radius == 0:
case Triangle t when t.Base == 0 || t.Height == 0:
case Rectangle r when r.Length == 0 || r.Height == 0:
return 0;
case Square s:
return s.Side * s.Side;
case Circle c:
return c.Radius * c.Radius * Math.PI;
case Triangle t:
return t.Base * t.Height / 2;
case Rectangle r:
return r.Length * r.Height;
case null:
throw new ArgumentNullException(paramName: nameof(shape), message: "Shape must not be null");
default:
throw new ArgumentException(
message: "shape is not a recognized shape",
paramName: nameof(shape));
}
}
O comportamento especial para o padrão null é interessante porque a constante null no padrão não tem um
tipo, mas pode ser convertida em qualquer tipo de referência ou tipo que permite valor nulo. Em vez de converter
um null em qualquer tipo, a linguagem define que um null valor não será correspondente ao padrão de
qualquer tipo, independentemente do tipo de tempo de compilação da variável. Esse comportamento torna o
novo padrão de tipo baseado em switch consistente com a instrução is : instruções is sempre retornam
false quando o valor sendo verificado é null . Isso também é mais simples: depois de verificar o tipo, não é
necessário fazer uma verificação adicional de nulos. Você pode ver isso pelo fato de que não há nenhuma
verificação de nulos em nenhum dos blocos de casos dos exemplos acima: elas não são necessárias, já que a
correspondência do padrão de tipo assegura um valor não nulo.
NOTE
O compilador não emite um aviso nos casos em que um case default foi gravado, mas nunca será executado. Isso é
coerente com a atual comportamento da instrução switch , em que todos os casos possíveis foram listados.
A terceira regra introduz usos em que um case var pode ser útil. Imagine que você esteja fazendo uma
correspondência de padrão em que a entrada é uma cadeia de caracteres e você está pesquisando valores de
comando conhecidos. Você pode escrever algo parecido com:
case "square":
return new Square(4);
case "large-circle":
return new Circle(12);
O case var corresponde a null , a cadeia de caracteres vazia ou qualquer cadeia de caracteres que contém
somente espaços em branco. Observe que o código anterior usa o operador ?. para garantir que ele não gere
um NullReferenceExceptionacidentalmente. O case default lida com outros valores de cadeia de caracteres que
não são entendidos pelo analisador de comando.
Este é um exemplo em que talvez você queira considerar uma expressão case var diferente de uma expressão
default .
Conclusões
Os constructos de correspondência de padrões permitem que você gerencie facilmente o fluxo de controle entre
diferentes tipos e variáveis que não são relacionadas por uma hierarquia de herança. Você também pode
controlar a lógica para usar qualquer condição testada na variável. Ela habilita os padrões e expressões que você
precisará com mais frequência conforme cria aplicativos mais distribuídos, no qual os dados e os métodos que
manipulam esses dados são separados. Você observará que os structs de forma usados nesse exemplo não
contêm nenhum método, apenas propriedades somente leitura. A correspondência de padrões funciona com
qualquer tipo de dados. Você escreve expressões que examinam o objeto e toma decisões de fluxo de controle
com base nessas condições.
Compare o código deste exemplo com o design que viria após a criação de uma hierarquia de classe para um
resumo de Shape e formas derivadas específicas, cada uma com sua própria implementação de um método
virtual para calcular a área. Muitas vezes você descobrirá que as expressões de correspondência de padrões
podem ser uma ferramenta muito útil quando estiver trabalhando com os dados e desejar separar as
preocupações de armazenamento de dados das preocupações de comportamento.
Escrever um código C# seguro e eficiente
30/10/2019 • 30 minutes to read • Edit Online
Novos recursos em C# permitem escrever código seguro verificável com melhor desempenho. Se você aplicar
essas técnicas com cuidado, menos cenários exigirão código não seguro. Esses recursos tornam fácil usar
referências a tipos de valor como argumentos e retornos de método. Quando realizadas com segurança, essas
técnicas minimizam a cópia de tipos de valor. Usando tipos de valor, é possível minimizar a quantidade de
alocações e passagens de coleta de lixo.
Grande parte do código de exemplo neste artigo usa recursos adicionados no C# 7.2. Para usar esses recursos, é
necessário configurar seu projeto para usar o C# 7.2 ou posterior. Para saber mais sobre como definir a versão da
linguagem, confira configure the language version (configurar a versão da linguagem).
Este artigo se concentra em técnicas para o gerenciamento eficiente de recursos. Uma vantagem de usar tipos de
valor é que eles geralmente evitam alocações de heap. A desvantagem é que eles são copiados por valor. Essa
compensação dificulta a otimização de algoritmos que operam em grandes quantidades de dados. Os novos
recursos de linguagem no C# 7.2 oferecem mecanismos que habilitam o código eficiente seguro que usa
referências para tipos de valor. Use esses recursos criteriosamente para minimizar tanto as alocações quanto as
operações de cópia. Este artigo explora esses novos recursos.
Este artigo se concentra nas seguintes técnicas de gerenciamento de recursos:
declarar uma readonly struct para expressar que um tipo é imutável e permite que o compilador salve
cópias ao usar os parâmetros in .
Se um tipo não puder ser imutável, declare struct Membros readonly para indicar que o membro não
modifica o estado.
usar um retorno ref readonly quando o valor retornado é struct maior que IntPtr.Size e o tempo de vida de
armazenamento é maior que o método que retorna o valor.
Quando o tamanho de um readonly struct é maior que IntPtr.Size, você deve passá-lo como um parâmetro
in por motivos de desempenho.
Nunca passe um struct como um parâmetro in , a menos que seja declarado com o modificador readonly
ou o método chame somente membros de readonly da estrutura. A violação dessas diretrizes pode afetar
negativamente o desempenho e pode levar a um comportamento obscuro.
Use um ref struct ou um readonly ref struct como Span<T> ou ReadOnlySpan<T> para trabalhar com a
memória como uma sequência de bytes.
Essas técnicas forçam você a equilibrar duas metras concorrentes em relação às referências e aos valores. As
variáveis que são tipos de referência mantêm uma referência ao local na memória. As variáveis que são tipos de
valor contêm diretamente seu valor. Essas diferenças realçam as principais diferenças que são importantes para
gerenciar recursos de memória. Normalmente, os tipos de valor são copiados quando passados para um método
ou retornados de um método. Esse comportamento inclui copiar o valor do this ao chamar membros de um
tipo de valor. O custo da cópia está relacionado ao tamanho do tipo. Os tipos de referência são alocados no heap
gerenciado. Cada novo objeto requer uma nova alocação e, subsequentemente, deve ser recuperado. Ambas as
operações levam tempo. A referência é copiada quando um tipo de referência é passado como um argumento
para um método ou retornado de um método.
Este artigo usa o seguinte exemplo de conceito da estrutura de ponto 3D para explicar estas recomendações:
public struct Point3D
{
public double X;
public double Y;
public double Z;
}
Siga esta recomendação sempre que sua intenção de design for criar um tipo de valor imutável. Quaisquer
melhorias no desempenho são um benefício adicional. O readonly struct expressa claramente a intenção do
design.
O exemplo anterior mostra muitos dos locais em que você pode aplicar o modificador de readonly : métodos,
propriedades e acessadores de propriedade. Se você usar Propriedades autoimplementadas, o compilador
adicionará o modificador de readonly ao acessador de get para propriedades de leitura/gravação. O compilador
adiciona o modificador de readonly às declarações de propriedade implementadas automaticamente para
propriedades com apenas um acessador get .
Adicionar o modificador de readonly a membros que não permutam o estado fornece dois benefícios
relacionados. Primeiro, o compilador impõe sua intenção. Esse membro não pode mutar o estado da estrutura
nem pode acessar um membro que também não esteja marcado readonly . Em segundo lugar, o compilador não
criará cópias defensivas de parâmetros de in ao acessar um membro de readonly . O compilador pode tornar
essa otimização segura porque garante que a struct não seja modificada por um membro de readonly .
No entanto, a seguinte definição de propriedade pode ser retornada por referência, porque o valor retornado é
um membro estático:
Você não deseja chamadores que modificam a origem, então deve retornar o valor por readonly ref :
Retornar ref readonly permite que você salvar a cópia de estruturas maiores e preserve a imutabilidade de seus
membros de dados internos.
No site de chamada, os chamadores fazem a opção de usar a propriedade Origin como um readonly ref ou
como um valor:
A primeira atribuição no código anterior faz uma cópia da constante Origin e atribui essa cópia. A segunda
atribui uma referência. Observe que o modificador readonly deve ser parte da declaração da variável. A
referência à qual ele se relaciona não pode ser modificada. As tentativas de modificá-la resultam em um erro em
tempo de compilação.
O modificador readonly é necessário na declaração de originReference .
O compilador impõe que o autor da chamada não pode modificar a referência. As tentativas de atribuir o valor
diretamente geram um erro em tempo de compilação. No entanto, o compilador não pode saber se algum
método de membro modifica o estado do struct. Para garantir que o objeto não será modificado, o compilador
cria uma cópia e chama as referências de membro usando essa cópia. Todas as modificações são para essa cópia
de defesa.
Adicione o modificador in para passar um argumento por referência e declare que sua intenção de design é
passar argumentos por referência para evitar cópias desnecessárias. Você não pretende modificar o objeto usado
como esse argumento.
Essa prática geralmente melhora o desempenho para tipos de valor somente leitura que são maiores que
IntPtr.Size. Para tipos simples ( sbyte , byte , short , ushort , int , uint , long , ulong , char , float , double ,
decimal e bool , e tipos enum ), eventuais ganhos no desempenho são mínimos. Na verdade, o desempenho
pode ser degradado usando a passagem por referência para tipos menores que IntPtr.Size.
O código a seguir mostra um exemplo de um método que calcula a distância entre dois pontos no espaço 3D.
Os argumentos são duas estruturas que contêm três duplas. Uma dupla tem 8 bytes. Então, cada argumento tem
24 bytes. Ao especificar o modificador in , você passa uma referência de 4 ou 8 bytes para esses argumentos,
dependendo da arquitetura do computador. A diferença no tamanho é pequena, mas ela aumenta rapidamente
quando o aplicativo chama esse método em um loop estreito, usando muitos valores diferentes.
O modificador in complementa out e ref de outras formas também. Não é possível criar sobrecargas de um
método que diferem somente na presença de in , out ou ref . Essas novas regras apresentam o mesmo
comportamento que sempre foi definido para os parâmetros out e ref . Como os modificadores out e ref , os
tipos de valor não estão demarcados, porque o modificador in é aplicado.
O modificador in pode ser aplicado a qualquer membro que usa os seguintes parâmetros: métodos, delegados,
lambdas, funções locais, indexadores e operadores.
Outro recurso dos parâmetros in é que você pode usar valores literais ou constantes para o argumento para um
parâmetro in . Além disso, ao contrário de um parâmetro ref ou out , você não precisa aplicar o modificador
in no local da chamada. O código a seguir mostra dois exemplos para chamar o método CalculateDistance . O
primeiro usa duas variáveis locais transmitidas por referência. O segundo inclui uma variável temporária criada
como parte da chamada de método.
Há várias maneiras pelas quais um compilador impõe a natureza somente leitura de um argumento in . Em
primeiro lugar, o método chamado não pode ser atribuído diretamente a um parâmetro in . Não é possível
atribuí-lo diretamente a nenhum campo de um parâmetro in quando esse valor é um tipo struct . Além disso,
não é possível passar um parâmetro in para nenhum método usando o modificador ref ou out . Essas regras
se aplicam a qualquer campo de um parâmetro in , considerando que o campo seja um tipo struct e o
parâmetro também seja um tipo struct . Na verdade, essas regras são aplicadas a várias camadas de acesso de
membro, considerando que os tipos, em todos os níveis de acesso de membro, sejam structs . O compilador
impõe que os tipos struct passados como argumentos in e seus membros struct sejam variáveis somente
leitura quando usados como argumentos para outros métodos.
O uso de parâmetros in pode evitar os possíveis custos de desempenho com a realização de cópias. Isso não
altera a semântica de nenhuma chamada de método. Portanto, não é necessário especificar o modificador in no
site de chamada. A omissão do modificador in no site de chamada informa ao compilador que ele tem
permissão para fazer uma cópia do argumento pelos seguintes motivos:
Existe uma conversão implícita, mas não uma conversão de identidade do tipo de argumento no tipo de
parâmetro.
O argumento é uma expressão, mas não tem uma variável de armazenamento conhecida.
Há uma sobrecarga que é distinguível pela presença ou ausência de in . Nesse caso, a sobrecarga pelo valor é
uma correspondência melhor.
Essas regras são úteis conforme você atualiza o código existente para usar argumentos de referência somente
leitura. Dentro do método chamado, você pode chamar qualquer método de instância que use parâmetros por
valor. Nessas instâncias, uma cópia do parâmetro in é criada. Uma vez que o compilador pode criar uma variável
temporária para qualquer parâmetro in , você também pode especificar valores padrão para qualquer parâmetro
in . O código a seguir especifica a origem (ponto 0,0 ) como o valor padrão para o segundo ponto:
Para forçar o compilador a passar argumentos somente leitura por referência, especifique o modificador in nos
argumentos no site de chamada, conforme mostrado no código a seguir:
Esse comportamento facilita a adoção de parâmetros in ao longo do tempo nas grandes bases de código em
que os ganhos de desempenho são possíveis. Primeiro você adiciona o modificador in às assinaturas de método.
Em seguida, você adiciona o modificador in em sites de chamada e cria tipos readonly struct para que o
compilador evite criar cópias de defesa de parâmetros in em mais locais.
A designação do parâmetro in também pode ser usada com tipos de referência ou valores numéricos. No
entanto, os benefícios em ambos os casos serão mínimos, se houver.
A estrutura Point3D não é um struct somente leitura. Há seis chamadas de acesso de propriedade diferentes no
corpo deste método. No primeiro exame, você pode ter achado que esses acessos estavam seguros. No fim das
contas, um acessador get não deve modificar o estado do objeto. Mas não há nenhuma regra de linguagem que
impõe isso. É apenas uma convenção comum. Qualquer tipo pode implementar um acessador get que modificou
o estado interno. Sem alguma garantia de linguagem, o compilador deve criar uma cópia temporária do
argumento antes de chamar qualquer membro. O armazenamento temporário é criado na pilha, os valores do
argumento são copiados para o armazenamento temporário e o valor é copiado para a pilha para cada acesso de
membro como o argumento this . Em muitas situações, essas cópias prejudicam tanto o desempenho que a
passagem por valor é mais rápida do que a passagem por referência somente leitura quando o tipo de argumento
não é um readonly struct .
Em vez disso, se o cálculo de distância usar a struct imutável, ReadonlyPoint3D , os objetos temporários não serão
necessários:
O compilador gera um código mais eficiente quando você chama os membros de um readonly struct : a
referência this , em vez de uma cópia do receptor, é sempre um parâmetro in passado por referência ao
método membro. Essa otimização economiza cópias quando você usa um readonly struct como um argumento
in .
Você não deve passar um tipo de valor anulável como um argumento in . O tipo de Nullable<T> não é declarado
como uma struct somente leitura. Isso significa que o compilador deve gerar cópias defensivas para qualquer
argumento de tipo de valor anulável passado para um método usando o modificador in na declaração de
parâmetro.
Você pode ver um programa de exemplo que demonstra as diferenças de desempenho usando o
BenchmarkDotNet em nosso repositório de exemplos no github. Ele compara a passagem de um struct mutável
por valor e por referência com a passagem de um struct imutável por valor e por referência. O uso do struct
imutável e da passagem por referência é mais rápido.
Conclusões
Usar tipos de valor minimiza o número de operações de alocação:
o armazenamento para tipos de valor é alocado em pilhas para variáveis locais e argumentos de método.
o armazenamento para tipos de valor que são membros de outros objetos é alocado como parte desse objeto,
não como uma alocação separada.
o armazenamento para valores retornados de tipo de valor é alocado em pilhas.
Compare isso com tipos de referência nestas mesmas situações:
o armazenamento para tipos de referência é alocado em heap para variáveis locais e argumentos de método. A
referência é armazenada na pilha.
O armazenamento para tipos de referência que são membros de outros objetos são alocados separadamente
no heap. O objeto recipiente armazena a referência.
O armazenamento para valores retornados de tipo de referência é alocado em heap. A referência a esse
armazenamento é armazenada na pilha.
Minimizar alocações implica compensações. Você copia mais memória quando o tamanho do struct é maior que
o tamanho de uma referência. Normalmente, uma referência é 64 ou 32 bits e depende da CPU do computador
de destino.
Essas compensações geralmente têm o mínimo de impacto no desempenho. No entanto, para estruturas grandes
ou coleções maiores, o impacto no desempenho aumenta. O impacto pode ser grande em loops estreitos e em
afunilamentos para programas.
Esses aprimoramentos na linguagem C# são criados para algoritmos de desempenho críticos, nos quais as
alocações de memória são um importante fator para alcançar o desempenho necessário. Você pode achar que
geralmente não usa esses recursos no código que grava. No entanto, esses aprimoramentos foram adotados por
meio do .NET. À medida que cada vez mais APIs utilizam esses recursos, você verá o desempenho dos seus
aplicativos melhorar.
Consulte também
ref keyword
Retornos de ref e locais de ref
Árvores de expressão
30/10/2019 • 4 minutes to read • Edit Online
Se tiver usado o LINQ, você tem experiência com uma rica biblioteca em que os tipos Func fazem parte do
conjunto de API. (Se você não estiver familiarizado com o LINQ, provavelmente deseja ler o tutorial do LINQ e o
artigo sobre expressões lambda antes desta.) As árvores de expressão fornecem uma interação mais rica com os
argumentos que são funções.
Você escreve argumentos de função, normalmente usando expressões lambda, quando cria consultas LINQ. Em
uma consulta LINQ típica, esses argumentos de função são transformados em um delegado que o compilador
cria.
Quando quiser ter uma interação mais avançada, você precisa usar Árvores de expressão. Árvores de expressão
representam o código como uma estrutura que você pode examinar, modificar ou executar. Essas ferramentas
oferecem a capacidade de manipular o código em tempo de execução. Você pode escrever código que examina
algoritmos em execução ou injeta novos recursos. Em cenários mais avançados, você pode modificar algoritmos
em execução e até mesmo converter expressões C# para outro formato para execução em outro ambiente.
Provavelmente, você já escreveu código usando Árvores de expressão. APIs do LINQ do Entity Framework
aceitam Árvores de expressão como os argumentos para o padrão de expressão de consulta do LINQ. Isso
permite que o Entity Framework converta a consulta que você escreveu em C# em SQL, que é executado no
mecanismo do banco de dados. Outro exemplo é Moq, que é uma estrutura de simulação popular para .NET.
As seções restantes deste tutorial explorarão o que são as árvores de expressão, examinarão as classes de
estrutura que dão suporte a árvores de expressão e mostrarão como trabalhar com árvores de expressão. Você
aprenderá a ler árvores de expressão, criar árvores de expressão, criar árvores de expressão modificadas e
executar o código representado pelas árvores de expressão. Após a leitura, você estará pronto para usar essas
estruturas para criar algoritmos adaptáveis avançados.
1. Árvores de Expressão Explicadas
Compreender a estrutura e os conceitos por trás das Árvores de Expressão.
2. Tipos de Framework com Suporte a Árvores de Expressão
Saiba mais sobre as estruturas e classes que definem e manipulam as árvores de expressão.
3. Executando Expressões
Saiba como converter uma árvore de expressão representada como uma expressão lambda em um
delegado e como executar o delegado resultante.
4. Interpretando Expressões
Saiba como percorrer e examinar árvores de expressão para entender que código a árvore de expressão
representa.
5. Compilando Expressões
Saiba como construir os nós de uma árvore de expressão e compilar árvores de expressão.
6. Traduzindo Expressões
Saiba como compilar uma cópia modificada de uma árvore de expressão ou converter uma árvore de
expressão em um formato diferente.
7. Resumindo
Examine informações sobre as árvores de expressão.
Árvores de Expressão Explicadas
30/10/2019 • 9 minutes to read • Edit Online
var sum = 1 + 2;
Se você analisar isso como uma árvore de expressão, a árvore contém vários nós. O nó mais externo é uma
instrução de declaração de variável com atribuição ( var sum = 1 + 2; ). Esse nó mais externo contém vários nós
filho: uma declaração de variável, um operador de atribuição e uma expressão que representa o lado direito do
sinal de igual. Essa expressão é ainda subdividida em expressões que representam a operação de adição e os
operandos esquerdo e direito da adição.
Vamos detalhar um pouco mais as expressões que compõem o lado direito do sinal de igual. A expressão é 1 + 2 .
Essa é uma expressão binária. Mais especificamente, ela é uma expressão de adição binária. Uma expressão de
adição binária tem dois filhos, que representam os nós esquerdo e direito da expressão de adição. Aqui, ambos os
nós são expressões constantes: o operando esquerdo é o valor 1 e o operando direito é o valor 2 .
Visualmente, a declaração inteira é uma árvore: você pode começar no nó raiz e viajar até cada nó da árvore para
ver o código que constitui a instrução:
Instrução de declaração de variável com atribuição ( var sum = 1 + 2; )
Declaração de tipo de variável implícita ( var sum )
Palavra-chave var implícita ( var )
Declaração de nome de variável ( sum )
Operador de atribuição ( = )
Expressão de adição binária ( 1 + 2 )
Operando esquerdo ( 1 )
Operador de adição ( + )
Operando direito ( 2 )
Isso pode parecer complicado, mas é muito eficiente. Seguindo o mesmo processo, você pode decompor
expressões muito mais complicadas. Considere esta expressão:
A única coisa que você não pode fazer é modificar uma árvore de expressão. As árvores de expressão são
estruturas de dados imutáveis. Se quiser modificar (alterar) uma árvore de expressão, você deverá criar uma nova
árvore, que seja uma cópia da original, com as alterações desejadas.
Próximo – Tipos de estruturas que dão suporte às árvores de expressão
Tipos de Framework com suporte a árvores de
expressão
30/10/2019 • 5 minutes to read • Edit Online
if (addFive.NodeType == ExpressionType.Lambda)
{
var lambdaExp = (LambdaExpression)addFive;
Console.WriteLine(parameter.Name);
Console.WriteLine(parameter.Type);
}
Você pode ver neste exemplo simples que há muitos tipos envolvidos na criação e no trabalho com árvores de
expressão. Essa complexidade é necessária para fornecer os recursos do vocabulário avançado fornecido pela
linguagem C#.
Você encontrará mais ao examinar cada uma dessas três áreas. Invariavelmente, você encontrará o que precisa ao
começar com uma dessas três etapas.
Próximo – Executar árvores de expressão
Executar árvores de expressão
31/10/2019 • 11 minutes to read • Edit Online
Observe que o tipo delegado se baseia no tipo de expressão. Você deve conhecer o tipo de retorno e a lista de
argumentos se quiser usar o objeto delegado de maneira fortemente tipada. O método
LambdaExpression.Compile() retorna o tipo Delegate . Você precisará convertê-lo para o tipo delegado correto para
fazer com que as ferramentas de tempo de compilação verifiquem a lista de argumentos ou o tipo de retorno.
Advertências
A compilação de uma expressão lambda para um delegado e invocar esse delegado é uma das operações mais
simples que você pode realizar com uma árvore de expressão. No entanto, mesmo com essa operação simples, há
limitações que você deve estar ciente.
As expressões lambda criam fechamentos sobre todas as variáveis locais que são referenciadas na expressão. Você
deve assegurar que todas as variáveis que farão parte do delegado são utilizáveis no local em que você chamar
Compile e no momento em que você executar o delegado resultante.
Em geral, o compilador garantirá que isso seja verdadeiro. No entanto, se sua expressão acessa uma variável que
implementa IDisposable , é possível que seu código descarte o objeto enquanto ele ainda é mantido pela árvore
de expressão.
Por exemplo, esse código funciona bem, porque int não implementa IDisposable :
O delegado capturou uma referência à variável local constant . Essa variável é acessada a qualquer momento
mais tarde, quando a função retornada por CreateBoundFunc for executada.
No entanto, considere essa classe (bastante artificial) que implementa IDisposable :
public class Resource : IDisposable
{
private bool isDisposed = false;
public int Argument
{
get
{
if (!isDisposed)
return 5;
else throw new ObjectDisposedException("Resource");
}
}
Ao usá-la em uma expressão, como mostrado abaixo, você obterá uma ObjectDisposedException ao executar o
código referenciado pela propriedade Resource.Argument :
O delegado retornado desse método fechou sobre o objeto constant , que foi descartado. (Foi descartado, porque
foi declarado em uma instrução using ).
Agora, ao executar o delegado retornado desse método, uma ObjectDisposedException será lançada no ponto de
execução.
Parece realmente estranho ter um erro de runtime que representa um constructo de tempo de compilação, mas
esse é o mundo em que entramos quando trabalhamos com árvores de expressão.
Há muitas permutações desse problema, portanto é difícil oferecer diretrizes gerais para evitá-lo. Tenha cuidado
ao acessar variáveis locais quando estiver definindo expressões e tenha cuidado ao acessar o estado no objeto
atual (representado por this ) quando estiver criando uma árvore de expressão que pode ser retornada por uma
API pública.
O código na sua expressão pode referenciar métodos ou propriedades em outros assemblies. Esse assembly deve
estar acessível quando a expressão for definida, quando ela for compilada e quando o delegado resultante for
chamado. Você vai se deparar com uma ReferencedAssemblyNotFoundException nos casos em que ele não estiver
presente.
Resumo
As árvores de expressão que representam expressões lambda podem ser compiladas para criar um delegado que
pode ser executado. Isso fornece um mecanismo para executar o código representado por uma árvore de
expressão.
A árvore de expressão representa o código que seria executado para qualquer constructo específico que você criar.
Contanto que o ambiente em que você compilar e executar o código corresponda ao ambiente em que você criar a
expressão, tudo funcionará conforme o esperado. Quando isso não acontece, os erros são bastante previsíveis e
serão capturados em seus primeiros testes de qualquer código usando as árvores de expressão.
Próximo – Interpretando expressões
Interpretando Expressões
30/10/2019 • 20 minutes to read • Edit Online
Agora, vamos escrever o código que examinaria essa expressão e escrever algumas propriedades importantes
sobre ele. Este é o código:
Não estou usando var para declarar essa árvore de expressão, pois isso não é possível porque o lado direito
da atribuição é de um tipo implícito. Para entender isso mais profundamente, leia isto.
O nó raiz é um LambdaExpression . Para obter o código interessante no lado direito do operador => , você precisa
encontrar um dos filhos de LambdaExpression . Faremos isso com todas as expressões nesta seção. O nó pai nos
ajudar a localizar o tipo de retorno do LambdaExpression .
Para examinar cada nó nesta expressão, precisaremos visitar recursivamente alguns nós. Esta é uma primeira
implementação simples:
Expression<Func<int, int, int>> addition = (a, b) => a + b;
Você vai notar que há muita repetição no exemplo de código acima. Vamos limpar tudo isso e criar um visitante de
nós de expressão com uma finalidade mais geral. Para isso, precisaremos escrever um algoritmo recursivo.
Qualquer nó poderia ser de um tipo que pode ter filhos. Qualquer nó que tem filhos exige que nós visitemos esses
filhos e determinemos o que é esse nó. Esta é a versão limpa que utiliza a recursão para visitar as operações de
adição:
// Lambda Visitor
public class LambdaVisitor : Visitor
{
private readonly LambdaExpression node;
public LambdaVisitor(LambdaExpression node) : base(node)
{
this.node = node;
}
// Parameter visitor:
public class ParameterVisitor : Visitor
{
private readonly ParameterExpression node;
public ParameterVisitor(ParameterExpression node) : base(node)
{
this.node = node;
}
public override void Visit(string prefix)
{
Console.WriteLine($"{prefix}This is an {NodeType} expression type");
Console.WriteLine($"{prefix}Type: {node.Type.ToString()}, Name: {node.Name}, ByRef: {node.IsByRef}");
}
}
// Constant visitor:
public class ConstantVisitor : Visitor
{
private readonly ConstantExpression node;
public ConstantVisitor(ConstantExpression node) : base(node)
{
this.node = node;
}
Esse algoritmo é a base de um algoritmo que pode visitar qualquer LambdaExpression arbitrário. Há várias lacunas,
uma delas é que o código que criei pesquisa somente por uma amostra muito pequena dos possíveis conjuntos de
nós de árvore de expressão que ele pode encontrar. No entanto, ainda é possível aprender bastante com o que ele
produz. (O caso padrão no método Visitor.CreateFromExpression imprime uma mensagem no console de erro
quando um novo tipo de nó é encontrado. Dessa forma, você sabe que precisa adicionar um novo tipo de
expressão.)
Quando executa esse visitante na expressão de adição mostrada acima, você obtém a saída a seguir:
Agora que criou uma implementação de visitante mais geral, você pode visitar e processar muitos tipos diferentes
de expressões.
Antes de executar isso no algoritmo de visitante, tente pensar no que poderia ser a saída. Lembre-se de que o
operador + é um operador binário: ele deve ter dois filhos, que representam os operandos esquerdo e direito. Há
várias maneiras possíveis de construir uma árvore que podem ser corretas:
Você pode ver a separação em duas possíveis respostas para realçar a mais promissora. A primeira representa
expressões associativas à direita. A segunda representa expressões associativas à esquerda. A vantagem desses
dois formatos é que o formato pode ser dimensionado para qualquer número arbitrário de expressões de adição.
Se executar essa expressão por meio do visitante, você verá essa saída, verificando se a expressão de adição
simples é associativa à esquerda.
Para executar esse exemplo e ver a árvore de expressão completa, eu precise fazer uma alteração na árvore de
expressão de origem. Quando a árvore de expressão contém todas as constantes, a árvore resultante contém
apenas o valor constante de 10 . O compilador executa toda a adição e reduz a expressão a sua forma mais
simples. Simplesmente adicionar uma variável à expressão é suficiente para ver a árvore original:
Crie um visitante para essa soma e execute o visitante; você verá esta saída:
Você também pode executar qualquer um dos outros exemplos pelo código visitante e ver que árvore ele
representa. Veja um exemplo da expressão sum3 acima (com um parâmetro adicional para impedir que o
compilador calcule a constante):
Expression<Func<int, int, int>> sum3 = (a, b) => (1 + a) + (3 + b);
Observe que os parênteses não fazem parte da saída. Não há nenhum nó na árvore de expressão que representa
os parênteses na expressão de entrada. A estrutura de árvore de expressão contém todas as informações
necessárias para comunicar a precedência.
Este código representa uma possível implementação da função fatorial matemática. A maneira como escrevi este
código destaca duas limitações da criação de árvores de expressão atribuindo expressões lambda a Expressões.
Primeiro, lambdas de instrução não são permitidos. Isso significa que eu não posso usar loops, blocos, instruções
if/else e outras estruturas de controle comuns em C#. Estou limitado ao uso de expressões. Em segundo lugar, não
posso chamar recursivamente a mesma expressão. Eu poderia se ela já fosse um delegado, mas não posso chamá-
la em sua forma de árvore de expressão. Na seção criando árvores de expressão, você aprenderá técnicas para
superar essas limitações.
Nesta expressão, você encontrará todos esses tipos de nós:
1. Igual (expressão binária)
2. Multiplicar (expressão binária)
3. Condicional (a expressão ? :)
4. Expressão de chamada de método (chamar Range() e Aggregate() )
Uma maneira de modificar o algoritmo do visitante é continuar executando-o e escrever o tipo de nó toda vez que
você atingir sua cláusula default . Após algumas iterações, você terá cisto todos os nós potenciais. Então, você
tem tudo de que você precisa. O resultado seria algo semelhante a:
Criando nós
Vamos começar de forma relativamente simples mais uma vez. Vamos usar a expressão de adição com que tenho
trabalhado durante essas seções:
Para construir essa árvore de expressão, você precisará criar os nós de folha. Os nós de folha são constantes, de
modo que você pode usar o método Expression.Constant para criar os nós:
Depois que tiver a expressão de adição, você pode criar a expressão lambda:
Essa é uma expressão lambda muito simples, pois não contém argumentos. Posteriormente nesta seção, você
verá como mapear argumentos para parâmetros e criar expressões mais complicadas.
Para expressões que são simples como essa, você pode combinar todas as chamadas em uma única instrução:
Em seguida, você precisa criar uma expressão de chamada de método para a chamada para Math.Sqrt .
Depois, por fim, você coloca a chamada de método em uma expressão lambda e define os argumentos para a
expressão lambda:
Neste exemplo mais complicado, você verá mais algumas técnicas de que frequentemente precisará para criar
árvores de expressão.
Primeiro, você precisa criar os objetos que representam parâmetros ou variáveis locais antes de usá-los. Após ter
criado esses objetos, você pode usá-los em sua árvore de expressão quando for necessário.
Depois, você precisa usar um subconjunto das APIs de reflexão para criar um objeto MethodInfo para que possa
criar uma árvore de expressão para acessar esse método. Você deve se limitar ao subconjunto das APIs de
reflexão que estão disponíveis na plataforma do .NET Core. Mais uma vez, essas técnicas se estenderão a outras
árvores de expressão.
Acima, observe que eu não criei a árvore de expressão, apenas o delegado. Usando a classe Expression , não é
possível criar lambdas de instrução. Este é o código que é necessário para criar a mesma funcionalidade. Ele é
complicado pelo fato de que não há uma API para criar um loop while . Em vez disso, você precisa criar um loop
que contém um teste condicional e um destino para o rótulo para interromper o loop.
O código para criar a árvore de expressão para a função fatorial é bem mais longo, mais complicado e está cheio
de rótulos e instruções de interrupção, bem como outros elementos que gostamos de evitar em nossas tarefas de
codificação cotidianas.
Para esta seção, também atualizei o código de visitante para visitar cada nó nessa árvore de expressão e gravar
informações sobre os nós que são criados neste exemplo. Você pode exibir ou baixar o código de exemplo no
repositório dotnet/docs do GitHub. Experimente por conta própria criando e executando os exemplos. Para obter
instruções de download, consulte Exemplos e tutoriais.
Examinando as APIs
As APIs de árvore de expressão são algumas das mais difíceis de navegar no .NET Core, mas não tem problema.
Sua finalidade é uma tarefa bastante complexa: escrever código que gera código em runtime. Eles são
necessariamente complicadas para fornecer um equilíbrio entre dar suporte a todas as estruturas de controle
disponíveis na linguagem C# e manter a área de superfície das APIs tão pequena quanto for razoável. Esse
equilíbrio significa que muitas estruturas de controle são representadas não por seus constructos em C#, mas por
constructos que representam a lógica subjacente que o compilador gera desses constructos de nível superior.
Além disso, no momento, há expressões de C# que não podem ser criadas diretamente usando os métodos de
classe Expression . Em geral, esses serão os operadores e expressões mais novos adicionadas no C# 5 e no C# 6.
(Por exemplo, expressões async não podem ser criadas e o novo operador ?. não pode ser criado
diretamente.)
Próximo – Traduzindo expressões
Movendo árvores de expressão
30/10/2019 • 10 minutes to read • Edit Online
Mover é visitar
O código que você compila para mover uma árvore de expressão é uma extensão do que você já viu para visitar
todos os nós em uma árvore. Quando você move uma árvore de expressão, visita todos os nós e ao visitá-los, cria
a nova árvore. A nova árvore pode conter referências aos nós originais ou aos novos nós que você colocou na
árvore.
Vamos ver isso em ação, visitando uma árvore de expressão e criando uma nova árvore com alguns nós de
substituição. Neste exemplo, vamos substituir qualquer constante com uma constante que seja dez vezes maior.
Caso contrário, vamos deixar a árvore de expressão intacta. Em vez de ler o valor da constante e substituí-la por
uma nova constante, faremos essa substituição através da troca do nó constante por um novo nó que executa a
multiplicação.
Aqui, quando você encontrar um nó constante, você criará um novo nó de multiplicação cujos filhos serão a
constante original e a constante 10 :
Ao substituir o nó original pelo substituto, uma nova árvore será formada, contendo as nossas modificações.
Podemos verificar isso compilando e executando a árvore substituída.
var one = Expression.Constant(1, typeof(int));
var two = Expression.Constant(2, typeof(int));
var addition = Expression.Add(one, two);
var sum = ReplaceNodes(addition);
var executableFunc = Expression.Lambda(sum);
A criação de uma nova árvore é uma combinação da visita aos nós da árvore existentes e a criação de novos nós,
inserindo-os na árvore.
Este exemplo mostra a importância das árvores de expressão serem imutáveis. Observe que a nova árvore criada
acima contém uma mistura de nós recém-criados e nós da árvore existente. E isso é seguro, porque os nós da
árvore existente não podem ser modificados. Isso pode resultar em eficiências significativas de memória. Os
mesmos nós podem ser usados em toda a árvore ou em várias árvores de expressão. Como os nós não podem
ser modificados, o mesmo nó pode ser reutilizado sempre que necessário.
Tem bastante código nisso, mas os conceitos são bastante acessíveis. Esse código visita filhos em uma pesquisa de
profundidade inicial. Ao encontrar um nó constante, o visitante retorna o valor da constante. Depois de visitar
ambos os filhos, os visitantes terão a soma que foi calculada para essa subárvore. Agora o nó de adição poderá
computar sua soma. Uma vez que todos os nós da árvore de expressão forem visitados, a soma será calculada.
Você pode executar o exemplo no depurador e rastrear a execução.
Vamos facilitar o rastreamento de como os nós são analisados e como a soma é calculada, percorrendo a árvore.
Esta é uma versão atualizada do método de agregação que inclui bastante informação de rastreamento:
private static int Aggregate(Expression exp)
{
if (exp.NodeType == ExpressionType.Constant)
{
var constantExp = (ConstantExpression)exp;
Console.Error.WriteLine($"Found Constant: {constantExp.Value}");
return (int)constantExp.Value;
}
else if (exp.NodeType == ExpressionType.Add)
{
var addExp = (BinaryExpression)exp;
Console.Error.WriteLine("Found Addition Expression");
Console.Error.WriteLine("Computing Left node");
var leftOperand = Aggregate(addExp.Left);
Console.Error.WriteLine($"Left is: {leftOperand}");
Console.Error.WriteLine("Computing Right node");
var rightOperand = Aggregate(addExp.Right);
Console.Error.WriteLine($"Right is: {rightOperand}");
var sum = leftOperand + rightOperand;
Console.Error.WriteLine($"Computed sum: {sum}");
return sum;
}
else throw new NotSupportedException("Haven't written this yet");
}
10
Found Addition Expression
Computing Left node
Found Addition Expression
Computing Left node
Found Constant: 1
Left is: 1
Computing Right node
Found Constant: 2
Right is: 2
Computed sum: 3
Left is: 3
Computing Right node
Found Addition Expression
Computing Left node
Found Constant: 3
Left is: 3
Computing Right node
Found Constant: 4
Right is: 4
Computed sum: 7
Right is: 7
Computed sum: 10
10
Rastreie a saída e acompanhe no código acima. Você será capaz de entender como o código visita cada nó e
calcula a soma, à medida que percorre a árvore e localiza a soma.
Agora, vejamos uma execução diferente, com a expressão fornecida por sum1 :
Embora a resposta final seja a mesma, a forma de percorrer a árvore é completamente diferente. Os nós são
percorridos em uma ordem diferente, porque a árvore foi construída com operações diferentes que ocorrem
primeiro.
Aprendendo mais
Este exemplo mostra um pequeno subconjunto do código que você compilaria para percorrer e interpretar os
algoritmos representados por uma árvore de expressão. Para obter uma discussão completa a respeito de todo o
trabalho necessário para compilar uma biblioteca de finalidade geral que move árvores de expressão para outra
linguagem, leia esta série escrita por Matt Warren. Ele entra em detalhes de como mover qualquer código que
você pode encontrar em uma árvore de expressão.
Espero que você tenha visto o verdadeiro poder das árvores de expressão. Você pode examinar um conjunto de
códigos, fazer as alterações que desejar nesse código e executar a versão modificada. Como as árvores de
expressão são imutáveis, você pode criar novas árvores usando os componentes de árvores existentes. Isso
minimiza a quantidade de memória necessária para criar árvores de expressão modificadas.
Próximo – Resumindo
Resumo de árvores de expressão
30/10/2019 • 2 minutes to read • Edit Online
Limitações
Há alguns elementos mais recentes da linguagem C# que não se convertem bem em árvores de expressão. As
árvores de expressão não podem conter expressões await ou expressões lambda async . Muitos dos recursos
adicionados na versão 6 do C# não aparecem exatamente como escritos nas árvores de expressão. Em vez disso,
os recursos mais recentes serão expostos em árvores de expressões na sintaxe anterior equivalente. Isso pode não
ser realmente uma limitação como você imagina. Na verdade, isso significa que é provável que o seu código, que
interpreta árvores de expressão, vai continuar funcionando da mesma forma que quando os novos recursos de
linguagem forem introduzidos.
Mesmo com essas limitações, as árvores de expressão permitem criar algoritmos dinâmicos que se apoiam na
interpretação e modificação do código que é representado como uma estrutura de dados. Elas são uma
ferramenta poderosa e é um dos recursos do ecossistema do .NET que habilita as avançadas bibliotecas, como o
Entity Framework, a realizarem o que fazem.
Interoperabilidade (Guia de Programação em C#)
04/11/2019 • 2 minutes to read • Edit Online
A interoperabilidade permite que você mantenha e aproveite os investimentos existentes em código não
gerenciado. O código que é executado sob o controle do CLR (Common Language Runtime) é chamado de código
gerenciado, e o código que é executado fora do CLR é chamado de código não gerenciado. COM, COM+,
componentes do C++, componentes do ActiveX e a API do Microsoft Windows são exemplos de código não
gerenciado.
O .NET Framework habilita a interoperabilidade com código não gerenciado por meio de serviços de invocação de
plataforma, o System.Runtime.InteropServices namespace, a interoperabilidade com C++ e a interoperabilidade
COM.
Nesta seção
Visão geral sobre interoperabilidade
Descreve métodos para fins de interoperabilidade entre código gerenciado em C# e código não gerenciado.
Como acessar objetos de interoperabilidade do Office usando recursos do Visual C#
Descreve os recursos que são introduzidos no Visual C# para facilitar a programação do Office.
Como usar propriedades indexadas na programação para interoperabilidade COM
Descreve como usar propriedades indexadas para acesso propriedades COM que têm parâmetros.
Como usar invocação de plataforma para executar um arquivo wave
Descreve como usar os serviços de invocação de plataforma para reproduzir um arquivo de som .wav no sistema
operacional Windows.
Passo a passo: programação do Office
Mostra como criar uma planilha do Excel e um documento do Word com um link para a planilha.
Exemplo de classe COM
Demonstra como expor uma classe C# como um objeto COM.
Especificação da Linguagem C#
Para obter mais informações, veja Noções básicas na Especificação da linguagem C#. A especificação da linguagem
é a fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Marshal.ReleaseComObject
Guia de Programação em C#
Interoperação com código não gerenciado
Passo a passo: programação do Office
Documentando seu código com comentários em
XML
25/11/2019 • 36 minutes to read • Edit Online
Comentários em documentação XML são um tipo especial de comentário, adicionados acima da definição de
qualquer membro ou tipo definido pelo usuário. Eles são especiais porque podem ser processados pelo
compilador para gerar um arquivo de documentação XML em tempo de compilação. O arquivo XML gerado pelo
compilador pode ser distribuído em conjunto com seu assembly .NET para que o Visual Studio e outros IDEs
possam usar o IntelliSense para mostrar informações rápidas sobre os tipos ou membros. Além disso, o arquivo
XML pode ser executado por ferramentas como DocFX e Sandcastle para gerar sites de referência de API.
Comentários de documentação XML, como todos os outros comentários, são ignorados pelo compilador.
É possível gerar o arquivo XML em tempo de compilação seguindo um destes procedimentos:
Se você estiver desenvolvendo um aplicativo com o .NET Core na linha de comando, poderá adicionar um
elemento GenerateDocumentationFile à seção <PropertyGroup> do arquivo de projeto. csproj. Você também
pode especificar o caminho para o arquivo de documentação diretamente usando DocumentationFile
elemento. O seguinte exemplo gera um arquivo XML no diretório do projeto com o mesmo nome de
arquivo raiz do assembly:
<GenerateDocumentationFile>true</GenerateDocumentationFile>
<DocumentationFile>bin\$(Configuration)\$(TargetFramework)\$(AssemblyName).xml</DocumentationFile>
Se estiver desenvolvendo um aplicativo usando o Visual Studio, clique com botão direito do mouse no
projeto e selecione Propriedades. Na caixa de diálogo Propriedades, selecione a guia Build e marque
Arquivo de documentação XML. Também é possível alterar o local em que o compilador grava o arquivo.
Se você estiver Compilando um aplicativo .NET Framework na linha de comando, adicione a opção-doc do
compilador ao compilar.
Comentários de documentação XML usam três barras ( /// ) e o corpo do comentário formatado em XML. Por
exemplo:
/// <summary>
/// This class does something.
/// </summary>
public class SomeClass
{
Passo a passo
Vamos examinar a documentação de uma biblioteca de matemática bastante básica para facilitar a
compreensão/contribuição por novos desenvolvedores e facilitar o uso por desenvolvedores de terceiros.
Este é o código para a biblioteca de matemática simples:
/*
The main Math class
Contains all methods for performing basic math functions
*/
public class Math
{
// Adds two integers and returns the result
public static int Add(int a, int b)
{
// If any parameter is equal to the max value of an integer
// and the other is greater than zero
if ((a == int.MaxValue && b > 0) || (b == int.MaxValue && a > 0))
throw new System.OverflowException();
return a + b;
}
return a + b;
}
A biblioteca de exemplo dá suporte a quatro operações aritméticas principais, add , subtract , multiply e divide ,
nos tipos de dados int e double .
Agora, você quer poder criar um documento de referência de API do seu código para desenvolvedores de terceiros
que usam sua biblioteca, mas não têm acesso ao código-fonte. Como já foi mencionado, as marcas da
documentação XML podem ser usadas para isso. Agora, você será apresentado às marcas XML padrão que têm
suporte do compilador de C#.
<summary>
A marca <summary> adiciona informações sucintas sobre um tipo ou membro. Vou demonstrar seu uso
adicionando-a à definição de classe Math e ao primeiro método Add . Fique à vontade para aplicá-la ao restante de
seu código.
/*
The main Math class
Contains all methods for performing basic math functions
*/
/// <summary>
/// The main Math class.
/// Contains all methods for performing basic math functions.
/// </summary>
public class Math
{
// Adds two integers and returns the result
/// <summary>
/// Adds two integers and returns the result.
/// </summary>
public static int Add(int a, int b)
{
// If any parameter is equal to the max value of an integer
// and the other is greater than zero
if ((a == int.MaxValue && b > 0) || (b == int.MaxValue && a > 0))
throw new System.OverflowException();
return a + b;
}
}
A marca <summary> é muito importante e é recomendável inclui-la, porque seu conteúdo é a principal fonte de
informações sobre o tipo ou membro no IntelliSense ou em um documento de referência de API.
<remarks>
A marca <remarks> complementa as informações sobre tipos ou membros que a marca <summary> fornece. Neste
exemplo, você apenas a adiciona à classe.
/*
The main Math class
Contains all methods for performing basic math functions
*/
/// <summary>
/// The main Math class.
/// Contains all methods for performing basic math functions.
/// </summary>
/// <remarks>
/// This class can add, subtract, multiply and divide.
/// </remarks>
public class Math
{
<returns>
A marca <returns> descreve o valor retornado de uma declaração de método. Assim como antes, o exemplo a
seguir ilustra a marca <returns> no primeiro método Add . É possível fazer o mesmo em outros métodos.
return a + b;
}
<valor>
A marca <value> é semelhante à marca <returns> , exceto pelo fato de você usá-la para propriedades. Supondo
que sua biblioteca Math tivesse uma propriedade estática chamada PI , você usaria essa marca desta forma:
/*
The main Math class
Contains all methods for performing basic math functions
*/
/// <summary>
/// The main Math class.
/// Contains all methods for performing basic math functions.
/// </summary>
/// <remarks>
/// This class can add, subtract, multiply and divide.
/// These operations can be performed on both integers and doubles
/// </remarks>
public class Math
{
/// <value>Gets the value of PI.</value>
public static double PI { get; }
}
<example>
Você usa a marca <example> para incluir um exemplo em sua documentação XML. Isso envolve o uso da marca
<code> filho.
return a + b;
}
A marca code preserva quebras de linha e recuos para exemplos mais longos.
<para>
Você usa a marca <para> para formatar o conteúdo dentro de sua marca pai. Normalmente, <para> é usado
dentro de uma marcação, como <remarks> ou <returns> , para dividir o texto em parágrafos. Você pode formatar
o conteúdo da marcação <remarks> para a definição de classe.
/*
The main Math class
Contains all methods for performing basic math functions
*/
/// <summary>
/// The main Math class.
/// Contains all methods for performing basic math functions.
/// </summary>
/// <remarks>
/// <para>This class can add, subtract, multiply and divide.</para>
/// <para>These operations can be performed on both integers and doubles.</para>
/// </remarks>
public class Math
{
<c>
Ainda com relação à formatação, use a marca <c> para marcar parte do texto como código. Ela é semelhante à
marca <code> , mas embutida. Ela é útil quando você deseja mostrar um exemplo de código rápido como parte do
conteúdo da marca. Vamos atualizar a documentação para a classe Math .
/*
The main Math class
Contains all methods for performing basic math functions
*/
/// <summary>
/// The main <c>Math</c> class.
/// Contains all methods for performing basic math functions.
/// </summary>
public class Math
{
<exception>
Usando a marca <exception> , você informa os desenvolvedores de que um método pode lançar exceções
específicas. Observando sua biblioteca Math , você pode ver que ambos os métodos Add lançarão uma exceção se
uma determinada condição for atendida. Algo menos óbvio, porém, é que o método inteiro Divide também será
lançado se o parâmetro b for zero. Agora, adicione a documentação de exceção a esse método.
/*
The main Math class
Contains all methods for performing basic math functions
*/
/// <summary>
/// The main <c>Math</c> class.
/// Contains all methods for performing basic math functions.
/// </summary>
public class Math
{
/// <summary>
/// Adds two integers and returns the result.
/// </summary>
/// <returns>
/// The sum of two integers.
/// </returns>
/// <example>
/// <code>
/// int c = Math.Add(4, 5);
/// if (c > 10)
/// {
/// Console.WriteLine(c);
/// }
/// </code>
/// </example>
/// <exception cref="System.OverflowException">Thrown when one parameter is max
/// and the other is greater than 0.</exception>
public static int Add(int a, int b)
{
if ((a == int.MaxValue && b > 0) || (b == int.MaxValue && a > 0))
throw new System.OverflowException();
return a + b;
}
/// <summary>
/// Adds two doubles and returns the result.
/// </summary>
/// <returns>
/// The sum of two doubles.
/// </returns>
/// <exception cref="System.OverflowException">Thrown when one parameter is max
/// <exception cref="System.OverflowException">Thrown when one parameter is max
/// and the other is greater than zero.</exception>
public static double Add(double a, double b)
{
if ((a == double.MaxValue && b > 0) || (b == double.MaxValue && a > 0))
throw new System.OverflowException();
return a + b;
}
/// <summary>
/// Divides an integer by another and returns the result.
/// </summary>
/// <returns>
/// The division of two integers.
/// </returns>
/// <exception cref="System.DivideByZeroException">Thrown when a division by zero occurs.</exception>
public static int Divide(int a, int b)
{
return a / b;
}
/// <summary>
/// Divides a double by another and returns the result.
/// </summary>
/// <returns>
/// The division of two doubles.
/// </returns>
/// <exception cref="System.DivideByZeroException">Thrown when a division by zero occurs.</exception>
public static double Divide(double a, double b)
{
return a / b;
}
}
O atributo cref representa uma referência a uma exceção que está disponível no ambiente de compilação atual.
Pode ser qualquer tipo definido no projeto ou um assembly referenciado. O compilador emitirá um aviso se o valor
não puder ser resolvido.
<see>
A marca <see> permite criar um link clicável para uma página de documentação para outro elemento de código.
Em nosso próximo exemplo, criaremos um link clicável entre os dois métodos Add .
/*
The main Math class
Contains all methods for performing basic math functions
*/
/// <summary>
/// The main <c>Math</c> class.
/// Contains all methods for performing basic math functions.
/// </summary>
public class Math
{
/// <summary>
/// Adds two integers and returns the result.
/// </summary>
/// <returns>
/// The sum of two integers.
/// </returns>
/// <example>
/// <code>
/// int c = Math.Add(4, 5);
/// if (c > 10)
/// {
/// Console.WriteLine(c);
/// }
/// </code>
/// </example>
/// <exception cref="System.OverflowException">Thrown when one parameter is max
/// and the other is greater than 0.</exception>
/// See <see cref="Math.Add(double, double)"/> to add doubles.
public static int Add(int a, int b)
{
if ((a == int.MaxValue && b > 0) || (b == int.MaxValue && a > 0))
throw new System.OverflowException();
return a + b;
}
/// <summary>
/// Adds two doubles and returns the result.
/// </summary>
/// <returns>
/// The sum of two doubles.
/// </returns>
/// <exception cref="System.OverflowException">Thrown when one parameter is max
/// and the other is greater than zero.</exception>
/// See <see cref="Math.Add(int, int)"/> to add integers.
public static double Add(double a, double b)
{
if ((a == double.MaxValue && b > 0) || (b == double.MaxValue && a > 0))
throw new System.OverflowException();
return a + b;
}
}
O cref é um atributo obrigatório que representa uma referência para um tipo ou seu membro que está
disponível no ambiente de compilação atual. Pode ser qualquer tipo definido no projeto ou um assembly
referenciado.
<seealso>
Você usa a marca <seealso> da mesma forma que usaria a marca <see> . A única diferença é que seu conteúdo
normalmente é colocado em uma seção "Consulte também". Aqui, adicionaremos uma marca seealso ao método
inteiro Add para fazer referência a outros métodos na classe que aceitam parâmetros inteiros:
/*
The main Math class
Contains all methods for performing basic math functions
*/
/// <summary>
/// The main <c>Math</c> class.
/// Contains all methods for performing basic math functions.
/// </summary>
public class Math
{
/// <summary>
/// Adds two integers and returns the result.
/// </summary>
/// <returns>
/// The sum of two integers.
/// </returns>
/// <example>
/// <code>
/// int c = Math.Add(4, 5);
/// if (c > 10)
/// {
/// Console.WriteLine(c);
/// }
/// </code>
/// </example>
/// <exception cref="System.OverflowException">Thrown when one parameter is max
/// and the other is greater than 0.</exception>
/// See <see cref="Math.Add(double, double)"/> to add doubles.
/// <seealso cref="Math.Subtract(int, int)"/>
/// <seealso cref="Math.Multiply(int, int)"/>
/// <seealso cref="Math.Divide(int, int)"/>
public static int Add(int a, int b)
{
if ((a == int.MaxValue && b > 0) || (b == int.MaxValue && a > 0))
throw new System.OverflowException();
return a + b;
}
}
O atributo cref representa uma referência para um tipo ou seu membro que está disponível no ambiente de
compilação atual. Pode ser qualquer tipo definido no projeto ou um assembly referenciado.
<param>
Você usa a marca <param> para descrever os parâmetros de um método. Este é um exemplo do método Add
duplo: o parâmetro que a marca descreve é especificado no atributo name obrigatório.
/*
The main Math class
Contains all methods for performing basic math functions
*/
/// <summary>
/// The main <c>Math</c> class.
/// Contains all methods for performing basic math functions.
/// </summary>
public class Math
{
/// <summary>
/// Adds two doubles and returns the result.
/// </summary>
/// <returns>
/// The sum of two doubles.
/// </returns>
/// <exception cref="System.OverflowException">Thrown when one parameter is max
/// and the other is greater than zero.</exception>
/// See <see cref="Math.Add(int, int)"/> to add integers.
/// <param name="a">A double precision number.</param>
/// <param name="b">A double precision number.</param>
public static double Add(double a, double b)
{
if ((a == double.MaxValue && b > 0) || (b == double.MaxValue && a > 0))
throw new System.OverflowException();
return a + b;
}
}
<typeparam>
Use a marca <typeparam> exatamente como a marca <param> , mas para declarações de método ou de tipo
genérico para descrever um parâmetro genérico. Adicione um método genérico rápido à sua classe Math para
verificar se uma quantidade é maior que outra.
/*
The main Math class
Contains all methods for performing basic math functions
*/
/// <summary>
/// The main <c>Math</c> class.
/// Contains all methods for performing basic math functions.
/// </summary>
public class Math
{
/// <summary>
/// Checks if an IComparable is greater than another.
/// </summary>
/// <typeparam name="T">A type that inherits from the IComparable interface.</typeparam>
public static bool GreaterThan<T>(T a, T b) where T : IComparable
{
return a.CompareTo(b) > 0;
}
}
<paramref>
Às vezes, você pode estar descrevendo o que um método faz, no que poderia ser uma marcação <summary> e
talvez queira fazer uma referência a um parâmetro. A marcação <paramref> é excelente para exatamente isso.
Vamos atualizar o resumo de nosso método Add de base dupla. Assim como a marca <param> , o nome do
parâmetro é especificado no atributo name obrigatório.
/*
The main Math class
Contains all methods for performing basic math functions
*/
/// <summary>
/// The main <c>Math</c> class.
/// Contains all methods for performing basic math functions.
/// </summary>
public class Math
{
/// <summary>
/// Adds two doubles <paramref name="a"/> and <paramref name="b"/> and returns the result.
/// </summary>
/// <returns>
/// The sum of two doubles.
/// </returns>
/// <exception cref="System.OverflowException">Thrown when one parameter is max
/// and the other is greater than zero.</exception>
/// See <see cref="Math.Add(int, int)"/> to add integers.
/// <param name="a">A double precision number.</param>
/// <param name="b">A double precision number.</param>
public static double Add(double a, double b)
{
if ((a == double.MaxValue && b > 0) || (b == double.MaxValue && a > 0))
throw new System.OverflowException();
return a + b;
}
}
<typeparamref>
Use a marca <typeparamref> exatamente como a marca <paramref> , mas para declarações de método ou de tipo
genérico para descrever um parâmetro genérico. É possível usar o mesmo método genérico criado anteriormente.
/*
The main Math class
Contains all methods for performing basic math functions
*/
/// <summary>
/// The main <c>Math</c> class.
/// Contains all methods for performing basic math functions.
/// </summary>
public class Math
{
/// <summary>
/// Checks if an IComparable <typeparamref name="T"/> is greater than another.
/// </summary>
/// <typeparam name="T">A type that inherits from the IComparable interface.</typeparam>
public static bool GreaterThan<T>(T a, T b) where T : IComparable
{
return a.CompareTo(b) > 0;
}
}
<list>
Use a marca <list> para formatar informações de documentação como uma lista ordenada, uma lista não
ordenada ou uma tabela. Crie uma lista não ordenada de cada operação matemática a que sua biblioteca Math dá
suporte.
/*
The main Math class
Contains all methods for performing basic math functions
*/
/// <summary>
/// The main <c>Math</c> class.
/// Contains all methods for performing basic math functions.
/// <list type="bullet">
/// <item>
/// <term>Add</term>
/// <description>Addition Operation</description>
/// </item>
/// <item>
/// <term>Subtract</term>
/// <description>Subtraction Operation</description>
/// </item>
/// <item>
/// <term>Multiply</term>
/// <description>Multiplication Operation</description>
/// </item>
/// <item>
/// <term>Divide</term>
/// <description>Division Operation</description>
/// </item>
/// </list>
/// </summary>
public class Math
{
É possível fazer uma lista ordenada ou tabela alterando o atributo type para number ou table , respectivamente.
Juntando as peças
Se você seguiu o tutorial e aplicou as marcas ao seu código quando necessário, seu código deve ser semelhante ao
seguinte:
/*
The main Math class
Contains all methods for performing basic math functions
*/
/// <summary>
/// The main <c>Math</c> class.
/// Contains all methods for performing basic math functions.
/// <list type="bullet">
/// <item>
/// <term>Add</term>
/// <description>Addition Operation</description>
/// </item>
/// <item>
/// <term>Subtract</term>
/// <description>Subtraction Operation</description>
/// </item>
/// <item>
/// <term>Multiply</term>
/// <description>Multiplication Operation</description>
/// </item>
/// <item>
/// <term>Divide</term>
/// <description>Division Operation</description>
/// </item>
/// </list>
/// </summary>
/// <remarks>
/// <para>This class can add, subtract, multiply and divide.</para>
/// <para>These operations can be performed on both integers and doubles.</para>
/// </remarks>
public class Math
{
// Adds two integers and returns the result
/// <summary>
/// Adds two integers <paramref name="a"/> and <paramref name="b"/> and returns the result.
/// </summary>
/// <returns>
/// The sum of two integers.
/// </returns>
/// <example>
/// <code>
/// int c = Math.Add(4, 5);
/// if (c > 10)
/// {
/// Console.WriteLine(c);
/// }
/// </code>
/// </example>
/// <exception cref="System.OverflowException">Thrown when one parameter is max
/// and the other is greater than 0.</exception>
/// See <see cref="Math.Add(double, double)"/> to add doubles.
/// <seealso cref="Math.Subtract(int, int)"/>
/// <seealso cref="Math.Multiply(int, int)"/>
/// <seealso cref="Math.Divide(int, int)"/>
/// <param name="a">An integer.</param>
/// <param name="b">An integer.</param>
public static int Add(int a, int b)
{
// If any parameter is equal to the max value of an integer
// and the other is greater than zero
if ((a == int.MaxValue && b > 0) || (b == int.MaxValue && a > 0))
throw new System.OverflowException();
return a + b;
}
return a + b;
}
No seu código, você pode gerar um site de documentação detalhada completo com referências cruzadas clicáveis.
Mas você se depara com outro problema: o código tornou-se difícil de ler. Há tanta informação para ser analisada
que será um pesadelo para qualquer desenvolvedor que desejar contribuir para esse código. Felizmente, há uma
marca XML que pode ajudá-lo a lidar com isso:
<include>
A marcação <include> permite fazer referência a comentários em um arquivo XML separado que descrevem os
tipos e membros em seu código-fonte, em vez de colocar comentários de documentação diretamente em seu
arquivo de código-fonte.
Agora, você moverá todas as suas marcas XML para um arquivo XML separado chamado docs.xml . Fique à
vontade para nomear o arquivo como desejar.
<docs>
<members name="math">
<Math>
<summary>
The main <c>Math</c> class.
Contains all methods for performing basic math functions.
</summary>
<remarks>
<para>This class can add, subtract, multiply and divide.</para>
<para>These operations can be performed on both integers and doubles.</para>
</remarks>
</Math>
<AddInt>
<summary>
Adds two integers <paramref name="a"/> and <paramref name="b"/> and returns the result.
Adds two integers <paramref name="a"/> and <paramref name="b"/> and returns the result.
</summary>
<returns>
The sum of two integers.
</returns>
<example>
<code>
int c = Math.Add(4, 5);
if (c > 10)
{
Console.WriteLine(c);
}
</code>
</example>
<exception cref="System.OverflowException">Thrown when one parameter is max
and the other is greater than 0.</exception>
See <see cref="Math.Add(double, double)"/> to add doubles.
<seealso cref="Math.Subtract(int, int)"/>
<seealso cref="Math.Multiply(int, int)"/>
<seealso cref="Math.Divide(int, int)"/>
<param name="a">An integer.</param>
<param name="b">An integer.</param>
</AddInt>
<AddDouble>
<summary>
Adds two doubles <paramref name="a"/> and <paramref name="b"/> and returns the result.
</summary>
<returns>
The sum of two doubles.
</returns>
<example>
<code>
double c = Math.Add(4.5, 5.4);
if (c > 10)
{
Console.WriteLine(c);
}
</code>
</example>
<exception cref="System.OverflowException">Thrown when one parameter is max
and the other is greater than 0.</exception>
See <see cref="Math.Add(int, int)"/> to add integers.
<seealso cref="Math.Subtract(double, double)"/>
<seealso cref="Math.Multiply(double, double)"/>
<seealso cref="Math.Divide(double, double)"/>
<param name="a">A double precision number.</param>
<param name="b">A double precision number.</param>
</AddDouble>
<SubtractInt>
<summary>
Subtracts <paramref name="b"/> from <paramref name="a"/> and returns the result.
</summary>
<returns>
The difference between two integers.
</returns>
<example>
<code>
int c = Math.Subtract(4, 5);
if (c > 1)
{
Console.WriteLine(c);
}
</code>
</example>
See <see cref="Math.Subtract(double, double)"/> to subtract doubles.
<seealso cref="Math.Add(int, int)"/>
<seealso cref="Math.Multiply(int, int)"/>
<seealso cref="Math.Divide(int, int)"/>
<param name="a">An integer.</param>
<param name="b">An integer.</param>
<param name="b">An integer.</param>
</SubtractInt>
<SubtractDouble>
<summary>
Subtracts a double <paramref name="b"/> from another double <paramref name="a"/> and returns the
result.
</summary>
<returns>
The difference between two doubles.
</returns>
<example>
<code>
double c = Math.Subtract(4.5, 5.4);
if (c > 1)
{
Console.WriteLine(c);
}
</code>
</example>
See <see cref="Math.Subtract(int, int)"/> to subtract integers.
<seealso cref="Math.Add(double, double)"/>
<seealso cref="Math.Multiply(double, double)"/>
<seealso cref="Math.Divide(double, double)"/>
<param name="a">A double precision number.</param>
<param name="b">A double precision number.</param>
</SubtractDouble>
<MultiplyInt>
<summary>
Multiplies two integers <paramref name="a"/> and <paramref name="b"/> and returns the result.
</summary>
<returns>
The product of two integers.
</returns>
<example>
<code>
int c = Math.Multiply(4, 5);
if (c > 100)
{
Console.WriteLine(c);
}
</code>
</example>
See <see cref="Math.Multiply(double, double)"/> to multiply doubles.
<seealso cref="Math.Add(int, int)"/>
<seealso cref="Math.Subtract(int, int)"/>
<seealso cref="Math.Divide(int, int)"/>
<param name="a">An integer.</param>
<param name="b">An integer.</param>
</MultiplyInt>
<MultiplyDouble>
<summary>
Multiplies two doubles <paramref name="a"/> and <paramref name="b"/> and returns the result.
</summary>
<returns>
The product of two doubles.
</returns>
<example>
<code>
double c = Math.Multiply(4.5, 5.4);
if (c > 100.0)
{
Console.WriteLine(c);
}
</code>
</example>
See <see cref="Math.Multiply(int, int)"/> to multiply integers.
<seealso cref="Math.Add(double, double)"/>
<seealso cref="Math.Subtract(double, double)"/>
<seealso cref="Math.Divide(double, double)"/>
<param name="a">A double precision number.</param>
<param name="a">A double precision number.</param>
<param name="b">A double precision number.</param>
</MultiplyDouble>
<DivideInt>
<summary>
Divides an integer <paramref name="a"/> by another integer <paramref name="b"/> and returns the
result.
</summary>
<returns>
The quotient of two integers.
</returns>
<example>
<code>
int c = Math.Divide(4, 5);
if (c > 1)
{
Console.WriteLine(c);
}
</code>
</example>
<exception cref="System.DivideByZeroException">Thrown when <paramref name="b"/> is equal to 0.
</exception>
See <see cref="Math.Divide(double, double)"/> to divide doubles.
<seealso cref="Math.Add(int, int)"/>
<seealso cref="Math.Subtract(int, int)"/>
<seealso cref="Math.Multiply(int, int)"/>
<param name="a">An integer dividend.</param>
<param name="b">An integer divisor.</param>
</DivideInt>
<DivideDouble>
<summary>
Divides a double <paramref name="a"/> by another double <paramref name="b"/> and returns the
result.
</summary>
<returns>
The quotient of two doubles.
</returns>
<example>
<code>
double c = Math.Divide(4.5, 5.4);
if (c > 1.0)
{
Console.WriteLine(c);
}
</code>
</example>
<exception cref="System.DivideByZeroException">Thrown when <paramref name="b"/> is equal to 0.
</exception>
See <see cref="Math.Divide(int, int)"/> to divide integers.
<seealso cref="Math.Add(double, double)"/>
<seealso cref="Math.Subtract(double, double)"/>
<seealso cref="Math.Multiply(double, double)"/>
<param name="a">A double precision dividend.</param>
<param name="b">A double precision divisor.</param>
</DivideDouble>
</members>
</docs>
No XML acima, os comentários de documentação de cada membro aparecem diretamente dentro de uma marca
cujo nome corresponde ao que eles fazem. Você pode escolher sua própria estratégia. Agora que você tem seus
comentários XML em um arquivo separado, vamos ver como seu código pode ficar mais legível usando a marca
<include> :
/*
The main Math class
Contains all methods for performing basic math functions
*/
*/
/// <include file='docs.xml' path='docs/members[@name="math"]/Math/*'/>
public class Math
{
// Adds two integers and returns the result
/// <include file='docs.xml' path='docs/members[@name="math"]/AddInt/*'/>
public static int Add(int a, int b)
{
// If any parameter is equal to the max value of an integer
// and the other is greater than zero
if ((a == int.MaxValue && b > 0) || (b == int.MaxValue && a > 0))
throw new System.OverflowException();
return a + b;
}
return a + b;
}
E aqui está: nosso código voltou a ser legível e nenhuma informação de documentação foi perdida.
O atributo file representa o nome do arquivo XML que contém a documentação.
O atributo path representa uma consulta XPath para o tag name presente no file especificado.
O atributo name representa o especificador de nome na marca que precede os comentários.
O atributo id que pode ser usado no lugar de name representa a ID da marca que precede os comentários.
Marcas definidas pelo usuário
Todas as marcas descritas acima representam as marcas que são reconhecidas pelo compilador C#. No entanto, o
usuário é livre para definir suas próprias marcas. Ferramentas como o Sandcastle dão suporte para marcas extras,
como <event> , <note> e até mesmo à documentação de namespaces. Ferramentas de geração de documentação
internas ou personalizadas também podem ser usadas com as marcas padrão e vários formatos de saída, de HTML
a PDF, podem ter suporte.
Recomendações
Documentar o código é recomendável por vários motivos. A seguir, temos algumas práticas recomendadas,
cenários de caso de uso gerais e coisas que você precisa saber ao usar marcas de documentação XML em seu
código C#.
Para fins de consistência, todos os tipos visíveis publicamente e seus membros devem ser documentados. Se
você precisar fazer isso, faça tudo.
Membros particulares também podem ser documentados usando comentários XML. No entanto, isso expõe o
funcionamento interno (potencialmente confidencial) de sua biblioteca.
No mínimo, os tipos e seus membros devem ter uma marca <summary> , porque seu conteúdo é necessário para
o IntelliSense.
O texto da documentação deve ser escrito usando frases terminadas com ponto final.
Classes parciais têm suporte total e informações da documentação serão concatenadas em uma única entrada
para esse tipo.
O compilador verifica a sintaxe das marcas <exception> , <include> , <param> , <see> , <seealso> e
<typeparam> .
O compilador valida os parâmetros que contêm caminhos de arquivo e referências para outras partes do
código.
Consulte também
Comentários de documentação XML (Guia de Programação em C#)
Marcas recomendadas para comentários de documentação (Guia de Programação em C#)
Controle de versão em C#
30/10/2019 • 11 minutes to read • Edit Online
Neste tutorial, você aprenderá o que significa o controle de versão no .NET. Você também aprenderá sobre os
fatores a serem considerados ao fazer o controle de versão de sua biblioteca, bem como ao atualizar para uma
nova versão de uma biblioteca.
Criando bibliotecas
Como um desenvolvedor que criou a bibliotecas .NET para uso público, provavelmente você esteve em situações
em que precisa distribuir novas atualizações. Como você realiza esse processo é muito importante, pois você
precisa garantir que haja uma transição suave do código existente para a nova versão da biblioteca. Aqui estão
Vários aspectos a considerar ao criar uma nova versão:
Controle de Versão Semântico
Controle de versão semântico (SemVer, de forma abreviada) é uma convenção de nomenclatura aplicada a versões
de sua biblioteca para indicar eventos com marcos específicos. Idealmente, as informações de versão que você
fornece a sua biblioteca devem ajudar os desenvolvedores a determinar a compatibilidade com seus projetos que
usam versões mais antigas da mesma biblioteca.
A abordagem mais básica ao SemVer é o formato de 3 componentes MAJOR.MINOR.PATCH , em que:
MAJOR é incrementado quando você faz alterações em APIs incompatíveis
MINOR é incrementado quando você adiciona funcionalidades de maneira compatível com versões anteriores
PATCH é incrementado quando você faz correções de bugs compatíveis com versões anteriores
Também há maneiras de especificar outros cenários, como versões de pré-lançamento etc. ao aplicar informações
de versão à sua biblioteca .NET.
Compatibilidade com versões anteriores
Conforme você lança novas versões de sua biblioteca, a compatibilidade com versões anteriores provavelmente
será uma de suas principais preocupações. Uma nova versão da biblioteca será compatível com a origem de uma
versão anterior se o código que depende da versão anterior puder, quando recompilado, trabalhar com a nova
versão. Uma nova versão da biblioteca será compatível de forma binária se um aplicativo que dependia da versão
anterior puder, sem recompilação, trabalhar com a nova versão.
Aqui estão algumas coisas a serem consideradas ao tentar manter a compatibilidade com versões mais antigas de
sua biblioteca:
Métodos virtuais: quando você torna um método em virtual não virtual na nova versão, significa que projetos
que substituem esse método precisarão ser atualizados. Essa é uma alteração muito grande e significativa que é
altamente desaconselhável.
Assinaturas de método: ao atualizar um comportamento de método exige que você altere sua assinatura
também, em vez disso, você deve criar uma sobrecarga para que o código que chama esse método ainda
funcione. Você sempre pode manipular a assinatura de método antiga para chamar a nova assinatura de
método para que a implementação permaneça consistente.
Atributo obsoleto: você pode usar esse atributo no seu código para especificar classes ou membros da classe
que foram preteridos e provavelmente serão removidos em versões futuras. Isso garante que os
desenvolvedores que utilizam sua biblioteca estarão melhor preparados para alterações significativas.
Argumentos de método opcionais: quando você tornar argumentos de método que antes eram opcionais em
compulsórios ou alterar seu valor padrão, todo código que não fornece esses argumentos precisará ser
atualizado.
NOTE
Tornar os argumentos compulsórios opcionais deve ter muito pouco efeito, especialmente se ele não alterar o
comportamento do método.
Quanto mais fácil for para os usuários atualizarem para a nova versão da sua biblioteca, mais provável será que
eles atualizem o quanto antes.
Arquivo de Configuração do Aplicativo
Como um desenvolvedor de .NET, há uma chance muito grande de você já ter encontrado o arquivo o app.config
na maioria dos tipos de projeto. Esse arquivo de configuração simples pode fazer muita diferença para melhorar a
distribuição de novas atualizações. Em geral, você deve criar suas bibliotecas de forma que as informações que
provavelmente forem alteradas regularmente sejam armazenadas no arquivo de app.config , dessa forma, quando
essas informações forem atualizadas, o arquivo de configuração de versões mais antigas precisará ser substituído
pelo novo sem a necessidade de recompilação da biblioteca.
Consumindo bibliotecas
Como um desenvolvedor que consome bibliotecas .NET criadas por outros desenvolvedores, vocês provavelmente
está ciente de que uma nova versão de uma biblioteca pode não ser totalmente compatível com seu projeto e pode
acabar precisando atualizar seu código para trabalhar com essas alterações.
Sorte para você, C# e o ecossistema .net vem com recursos e técnicas que nos permitem atualizar facilmente nosso
aplicativo para trabalhar com novas versões de bibliotecas que podem introduzir alterações significativas.
Redirecionamento de associação de assembly
Você pode usar o arquivo app. config para atualizar a versão de uma biblioteca que seu aplicativo usa. Ao adicionar
o que é chamado de redirecionamento de associação, você pode usar a nova versão da biblioteca sem precisar
recompilar seu aplicativo. O exemplo a seguir mostra como você atualizaria o arquivo app. config do aplicativo
para usar a versão 1.0.1 patch do ReferencedLibrary em vez da versão do 1.0.0 com a qual foi originalmente
compilado.
<dependentAssembly>
<assemblyIdentity name="ReferencedLibrary" publicKeyToken="32ab4ba45e0a69a1" culture="en-us" />
<bindingRedirect oldVersion="1.0.0" newVersion="1.0.1" />
</dependentAssembly>
NOTE
Essa abordagem só funcionará se a nova versão do ReferencedLibrary for compatível de forma binária com seu aplicativo.
Consulte a seção Compatibilidade com versões anteriores acima para ver as alterações importantes ao determinar a
compatibilidade.
new
Você usa o modificador new para ocultar membros herdados de uma classe base. Essa é uma maneira das classes
derivadas responderem a atualizações em classes base.
Veja o exemplo a seguir:
public class BaseClass
{
public void MyMethod()
{
Console.WriteLine("A base method");
}
}
b.MyMethod();
d.MyMethod();
}
Saída
A base method
A derived method
No exemplo acima, você pode ver como DerivedClass oculta o método MyMethod presente em BaseClass . Isso
significa que quando uma classe base na nova versão de uma biblioteca adiciona um membro que já existe em sua
classe derivada, você pode simplesmente usar o modificador new no membro de sua classe derivada para ocultar
o membro da classe base.
Quando nenhum modificador new é especificado, uma classe derivada ocultará por padrão membros conflitantes
em uma classe base e, embora um aviso do compilador seja gerado, o código ainda será compilado. Isso significa
que simplesmente adicionar novos membros a uma classe existente torna a nova versão da biblioteca compatível
com a origem e de forma binária com o código que depende dela.
override
O modificador override significa que uma implementação derivada estende a implementação de um membro da
classe base, em vez de ocultá-lo. O membro da classe base precisa ter o modificador virtual aplicado a ele.
public class MyBaseClass
{
public virtual string MethodOne()
{
return "Method One";
}
}
Saída
O modificador override é avaliado em tempo de compilação e o compilador gerará um erro se não encontrar um
membro virtual para substituir.
Seu conhecimento das técnicas discutidas e sua compreensão das situações em que elas serão usadas, será um
longo caminho para facilitar a transição entre as versões de uma biblioteca.
Instruções (C#)
08/11/2019 • 7 minutes to read • Edit Online
Na seção de Instruções do guia de C#, é possível encontrar respostas rápidas para perguntas comuns. Em alguns
casos, os artigos podem ser listados em várias seções. Queremos facilitar que sejam localizados por vários
caminhos de pesquisa.
Conceitos gerais de C#
Há vários truques e dicas que são práticas comuns do desenvolvedor de C#.
Inicializar objetos usando um inicializador de objeto.
Aprenda as diferenças entre passar um struct e uma classe para um método.
Use a sobrecarga de operador.
Implemente e chame um método de extensão personalizado.
Até os programadores de C# podem querer usar o My namespace do VB.
Crie um novo método para um tipo enum usando métodos de extensão.
Membros de classe e struct
As classes e os structs são criados para implementar seu programa. Essas técnicas são comumente usadas durante
a gravação de classes ou structs.
Declare propriedades implementadas automaticamente.
Declare e use propriedades de leitura/gravação.
Defina constantes.
Substitua o método ToString para fornecer saída de cadeia de caracteres.
Defina propriedades abstract.
Use os recursos de documentação para documentar seu código.
Implemente membros de interface explicitamente para manter a interface pública concisa.
Implemente membros de duas interfaces explicitamente.
Trabalhando com coleções
Esses artigos ajudam você a trabalhar com coleções de dados.
Inicialize um dicionário com um inicializador de coleção.
Tratamento de exceções
Programas .NET relatam que os métodos não concluíram seu trabalho com sucesso ao lançar exceções. Nesses
artigos, você aprenderá a trabalhar com exceções.
Trate exceções usando try e catch .
Limpe recursos usando as finally cláusulas.
Recupere com base em exceções não CLS (Common Language Specification).
Representantes e eventos
Representantes e eventos fornecem uma capacidade para estratégias que envolve blocos de código acoplados
livremente.
Declare, crie uma instância e use delegados.
Combine delegados multicast.
Os eventos fornecem um mecanismo para publicar ou assinar notificações.
Assine e cancele a assinatura de eventos.
Implemente eventos declarados nas interfaces.
Esteja em conformidade com as diretrizes do .NET Framework quando seu código publicar eventos.
Gere eventos definidos nas classes de base de classes derivadas.
Implemente acessadores de eventos personalizados.
Práticas do LINQ
O LINQ permite que você grave códigos para consultar qualquer fonte de dados compatível com o padrão de
expressão de consulta do LINQ. Esses artigos o ajudarão a entender o padrão e trabalhar com diferentes fontes de
dados.
Consulte uma coleção.
Use expressões lambda em uma consulta.
Use var nas expressões de consulta.
Retorne subconjuntos de propriedades de elementos em uma consulta.
Grave consultas com filtragem complexa.
Classifique os elementos de uma fonte de dados.
Classifique os elementos em múltiplas chaves.
Controle o tipo de uma projeção.
Conte as ocorrências de um valor em uma sequência de origem.
Calcule valores intermediários.
Mescle dados de várias fontes.
Encontre a diferença de conjunto entre duas sequências.
Depure resultados de consultas vazios.
Adicione métodos personalizados a consultas LINQ.
O método String.Split cria uma matriz de subcadeias, dividindo a cadeia de caracteres de entrada com base em
um ou mais delimitadores. Geralmente essa é a maneira mais fácil de separar uma cadeia de caracteres em limites
de palavra. Ele também é usado para dividir cadeias de caracteres em outros caracteres específicos ou cadeias de
caracteres.
NOTE
Os exemplos de C# neste artigo são executados no executador de código embutido Try.NET e no playground. Clique no
botão Executar para executar um exemplo em uma janela interativa. Ao executar o código, é possível modificá-lo e executar
o código modificado clicando em Executar novamente. O código modificado será executado na janela interativa ou, se a
compilação falhar, a janela interativa exibirá todos as mensagens de erro do compilador C#.
O código a seguir divide uma frase comum em uma matriz de cadeias de caracteres para cada palavra.
string phrase = "The quick brown fox jumps over the lazy dog.";
string[] words = phrase.Split(' ');
Cada instância de um caractere separador produz um valor na matriz retornada. Caracteres separadores
consecutivos produzem a cadeia de caracteres vazia como um valor na matriz retornada. Você pode ver isso no
exemplo a seguir, que usa espaço como um separador:
string phrase = "The quick brown fox jumps over the lazy dog.";
string[] words = phrase.Split(' ');
Esse comportamento facilita para formatos como arquivos CSV (valores separados por vírgula) que representam
dados de tabela. Vírgulas consecutivas representam uma coluna em branco.
Você pode passar um parâmetro StringSplitOptions.RemoveEmptyEntries opcional para excluir as cadeias de
caracteres vazias da matriz retornada. Para um processamento mais complicado da coleção retornada, você pode
usar o LINQ para manipular a sequência de resultado.
O String.Split pode usar vários caracteres separadores. O exemplo a seguir utiliza espaços, vírgulas, pontos, dois-
pontos e tabulações, todos passados em uma matriz que contém esses caracteres de separação para Split. O loop,
na parte inferior do código, exibe cada uma das palavras na matriz retornada.
char[] delimiterChars = { ' ', ',', '.', ':', '\t' };
As instâncias consecutivas de qualquer separador produzem a cadeia de caracteres vazia na matriz de saída:
O String.Split pode receber uma matriz de cadeias de caracteres (sequências de caracteres que atuam como
separadores para analisar a cadeia de caracteres de destino, em vez de um único caractere).
Você pode experimentar estes exemplos examinando o código em nosso repositório GitHub. Ou então, você pode
baixar os exemplos como um arquivo zip.
Consulte também
Guia de Programação em C#
Cadeias de Caracteres
Expressões regulares do .NET
Como concatenar várias cadeias deC# caracteres
(guia)
25/11/2019 • 6 minutes to read • Edit Online
Concatenação é o processo de acrescentar uma cadeia de caracteres ao final de outra cadeia de caracteres. Você
concatena cadeias de caracteres usando o operador + . Para literais de cadeia de caracteres e constantes de cadeia
de caracteres, a concatenação ocorre em tempo de compilação; não ocorre nenhuma concatenação de tempo de
execução. Para variáveis de cadeia de caracteres, a concatenação ocorre somente em tempo de execução.
NOTE
Os exemplos de C# neste artigo são executados no executador de código embutido Try.NET e no playground. Clique no
botão Executar para executar um exemplo em uma janela interativa. Ao executar o código, é possível modificá-lo e executar
o código modificado clicando em Executar novamente. O código modificado será executado na janela interativa ou, se a
compilação falhar, a janela interativa exibirá todos as mensagens de erro do compilador C#.
O exemplo a seguir usa a concatenação para dividir um literal de cadeia de caracteres longo em cadeias de
caracteres menores, a fim de melhorar a legibilidade no código-fonte. Essas partes são concatenadas em uma
única cadeia de caracteres em tempo de compilação. Não há custos de desempenho em tempo de execução,
independentemente da quantidade de cadeias de caracteres envolvidas.
System.Console.WriteLine(text);
Para concatenar variáveis de cadeia de caracteres, você pode usar os operadores + ou += , a interpolação de
cadeia de caracteres ou os métodos String.Format, String.Concat, String.Join ou StringBuilder.Append. O
operador + é fácil de usar e torna o código intuitivo. Mesmo ao usar vários operadores + em uma instrução, o
conteúdo da cadeia de caracteres será copiado apenas uma vez. O código a seguir mostra dois exemplos de como
usar os operadores + e += para concatenar cadeias de caracteres:
Em algumas expressões, é mais fácil concatenar cadeias de caracteres usando a interpolação de cadeia de
caracteres, conforme mostra o seguinte código:
NOTE
Em operações de concatenação de cadeia de caracteres, o compilador C# trata uma cadeia de caracteres nula da mesma
maneira que uma cadeia de caracteres vazia.
Outro método para concatenar cadeias de caracteres é o String.Format. Esse método funciona bem quando você
está criando uma cadeia de caracteres com base em um pequeno número de cadeias de caracteres de
componente.
Em outros casos, você pode combinar cadeias de caracteres em um loop em que você não sabe quantas cadeias
de caracteres de origem você está combinando, sendo que o número real de cadeias de caracteres de origem pode
ser grande demais. A classe StringBuilder foi projetada para esses cenários. O código a seguir usa o método
Append da classe StringBuilder para concatenar cadeias de caracteres.
Você pode ler mais sobre os motivos para escolher a concatenação de cadeia de caracteres ou a classe
StringBuilder
Outra opção para unir cadeias de caracteres de uma coleção é usar o método String.Concat. Use o método
String.Join se desejar separar as cadeias de caracteres de origem por um delimitador. O código a seguir combina
uma matriz de palavras usando os dois métodos:
string[] words = { "The", "quick", "brown", "fox", "jumps", "over", "the", "lazy", "dog." };
Por fim, você pode usar LINQ e o método Enumerable.Aggregate para unir cadeias de caracteres de uma coleção.
Esse método combina as cadeias de caracteres de origem usando uma expressão lambda. A expressão lambda faz
o trabalho de adicionar cada cadeia de caracteres ao acúmulo existente. O exemplo a seguir combina uma matriz
de palavras, adicionando um espaço entre cada palavra na matriz:
string[] words = { "The", "quick", "brown", "fox", "jumps", "over", "the", "lazy", "dog." };
Você pode experimentar estes exemplos examinando o código em nosso repositório GitHub. Ou então, você pode
baixar os exemplos como um arquivo zip.
Consulte também
String
StringBuilder
Guia de Programação em C#
Cadeias de Caracteres
Analisando Cadeias de Caracteres de Data e Hora no
.NET
31/10/2019 • 12 minutes to read • Edit Online
A análise de cadeias de caracteres para convertê-las em objetos DateTime exige que você especifique as
informações de como as datas e horas são representadas como texto. Diferentes culturas usam ordens diferentes
para dia, mês e ano. Algumas representações usam o relógio de 24 horas, outras especificam "AM" e "PM".
Alguns aplicativos precisam apenas da data. Outros precisam apenas da hora. Há também outros que precisam
especificar tanto a data quanto a hora. Os métodos que convertem cadeias de caracteres em objetos DateTime
permitem que você forneça informações detalhadas sobre os formatos esperados e os elementos de uma data e
hora que seu aplicativo precisa. Há três subtarefas para converter corretamente o texto em um DateTime:
1. Você deve especificar o formato esperado do texto que representa a data e hora.
2. Você pode especificar a cultura para o formato de data hora.
3. Você pode especificar como os componentes ausentes na representação de texto serão definidos na data e
hora.
Os métodos Parse e TryParse convertem muitas representações comuns de data e hora. Os métodos ParseExact e
TryParseExact convertem uma representação em cadeia de caracteres que cumpra o padrão especificado por uma
cadeia de caracteres de formato de data e hora. (Consulte os artigos sobre cadeias de caracteres de formato de
data e hora padrão e cadeias de caracteres de formato de data e hora personalizadas para maiores detalhes).
O objeto DateTimeFormatInfo atual fornece mais controle sobre como o texto deve ser interpretado como data e
hora. As propriedades de um DateTimeFormatInfo descrevem os separadores de data e hora e os nomes dos
meses, dias e eras, bem como o formato das designações "AM" e "PM". A cultura do thread atual fornece um
DateTimeFormatInfo que representa a cultura atual. Se você quer uma cultura específica ou configurações
personalizadas, é necessário especificar o parâmetro IFormatProvider de um método de análise. Para o parâmetro
IFormatProvider, especifique um objeto CultureInfo, o qual representa uma cultura ou um objeto
DateTimeFormatInfo.
O texto que representa uma data ou hora pode ter algumas informações ausentes. Por exemplo, a maioria das
pessoas supõe que a data "12 de março" representa o ano atual. Da mesma forma, "Março de 2018" representa o
mês de março do ano 2018. O texto que representa a hora geralmente só inclui horas, minutos e uma designação
de AM/PM. Os métodos de análise lidam com essas informações ausentes usando alguns padrões razoáveis:
Quando apenas a hora estiver presente, a parte de data usará a data atual.
Quando houver apenas uma data, a parte das horas será meia-noite.
Quando o ano não for especificado em uma data, o ano atual será usado.
Quando o dia do mês não for especificado, o primeiro dia do mês será usado.
Se a data estiver presente na cadeia de caracteres, ele deverá incluir o mês e, pelo menos, o dia ou o ano. Se a hora
estiver presente, ela deverá incluir a hora e os minutos ou o designador AM/PM.
Você pode especificar a constante NoCurrentDateDefault para substituir esses padrões. Ao usar essa constante, as
propriedades ausentes de ano, mês ou dia serão definidas com o valor 1 . O último exemplo usando Parse
demonstra esse comportamento.
Além de um componente de data e hora, a representação de cadeia de caracteres de data e hora pode incluir um
deslocamento que indique a diferença de tempo em relação ao UTC (Tempo Universal Coordenado). Por exemplo,
a cadeia de caracteres “2/14/2007 5:32:00 -7:00” define um horário sete horas antes do UTC. Se um deslocamento
for omitido da representação de cadeia de caracteres de um horário, a análise retornará um DateTime do objeto
com a propriedade Kind definida como DateTimeKind.Unspecified. Se um deslocamento for especificado, a análise
retorna um objeto DateTime com a propriedade Kind definida como DateTimeKind.Local e seu valor ajustado de
acordo com o fuso horário local do seu computador. É possível modificar esse comportamento usando um valor
DateTimeStyles com o método de análise.
O provedor de formato também é usado para interpretar uma data numérica ambígua. Não está claro quais
componentes da data representada pela cadeia de caracteres "02/03/04" são o mês, o dia e o ano. Os
componentes são interpretados de acordo com a ordem dos formatos de data semelhantes no provedor de
formato.
Parse
O exemplo a seguir ilustra o uso do método DateTime.Parse para converter uma string em um DateTime. Este
exemplo usa a cultura associada com o thread atual. Se o CultureInfo associado à cultura atual não puder analisar
a cadeia de caracteres de entrada, um FormatException será gerado.
TIP
Todos os exemplos de C# neste artigo são executados no navegador. Pressione o botão Executar para ver a saída. Você
também pode editá-los para experimentar como quiser.
NOTE
Esses exemplos estão disponíveis no repositório de documentos do GitHub para C# e VB. Ou você pode baixar o projeto
como um arquivo zip para C# ou VB.
É possível definir explicitamente a cultura cujas convenções de formatação serão usadas ao analisar uma cadeia de
caracteres. Você especifica um dos objetos DateTimeFormatInfo padrão retornados pela propriedade
CultureInfo.DateTimeFormat. O exemplo seguir usa um provedor de formato para analisar uma cadeia de
caracteres em alemão em um DateTime. Ele cria um CultureInfo que representa a cultura de-DE . Esse objeto
CultureInfo garante o sucesso da análise dessa cadeia de caracteres específica. Isso impede qualquer
configuração que está no CurrentCulture do CurrentThread.
No entanto, apesar de ser possível usar sobrecargas do método Parse para especificar provedores de formato
personalizados, o método não é compatível com a análise de formatos não padrão. Para analisar data e hora
expressadas em um formato não padrão, use o método ParseExact.
O exemplo a seguir usa a enumeração DateTimeStyles para especificar que as informações de data e hora atuais
não devem ser adicionadas ao DateTime nos campos não especificados.
ParseExact
O método DateTime.ParseExact converte uma cadeia de caracteres em um objeto DateTime se ele estiver em
conformidade com um dos padrões da cadeia de caracteres especificada. Quando uma cadeia de caracteres que
não é de uma das formas especificadas é passada para esse método, é lançada uma FormatException. Você pode
especificar um dos especificadores de formato de data e hora padrão ou uma combinação dos especificadores de
formato personalizados. Usando os especificadores de formato personalizados, é possível construir uma cadeia de
caracteres de reconhecimento personalizada. Para ver uma explicação dos especificadores, consulte os tópicos
cadeias de caracteres de formato de data e hora padrão e cadeias de caracteres de formato de data e hora
personalizadas.
No exemplo a seguir, o método DateTime.ParseExact recebe um objeto de cadeia de caracteres para analisar,
seguido por um especificador de formato, seguido por um objeto CultureInfo. Esse método ParseExact só
consegue analisar cadeias de caracteres que seguem o padrão de data por extenso na cultura en-US .
CultureInfo MyCultureInfo = new CultureInfo("en-US");
string[] MyString = { " Friday, April 10, 2009", "Friday, April 10, 2009" };
foreach (string dateString in MyString)
{
try
{
DateTime MyDateTime = DateTime.ParseExact(dateString, "D", MyCultureInfo);
Console.WriteLine(MyDateTime);
}
catch (FormatException)
{
Console.WriteLine("Unable to parse '{0}'", dateString);
}
}
// The example displays the following output:
// Unable to parse ' Friday, April 10, 2009'
// 4/10/2009 12:00:00 AM
Cada sobrecarga dos métodos Parse e ParseExact também tem um parâmetro IFormatProvider que oferece
informações específicas da cultura sobre a formatação da cadeia de caracteres. Esse objeto IFormatProvider é um
objeto CultureInfo que representa uma cultura padrão ou um objeto DateTimeFormatInfo que é retornado pela
propriedade CultureInfo.DateTimeFormat. O ParseExact também usa uma cadeia de caracteres adicional ou um
argumento de matriz de cadeia de caracteres que define um ou mais formatos de data e hora personalizados.
Consulte também
Análise de cadeias de caracteres
Formatando Tipos
Conversão de tipo no .NET
Formatos de data e hora padrão
Cadeias de caracteres de formato de data e hora personalizado
Como Pesquisar cadeias de caracteres
25/11/2019 • 7 minutes to read • Edit Online
Você pode usar duas estratégias principais para pesquisar texto em cadeias de caracteres. Os métodos da classe
String pesquisam por um texto específico. Expressões regulares pesquisam por padrões no texto.
NOTE
Os exemplos de C# neste artigo são executados no executador de código embutido Try.NET e no playground. Clique no
botão Executar para executar um exemplo em uma janela interativa. Ao executar o código, é possível modificá-lo e executar
o código modificado clicando em Executar novamente. O código modificado será executado na janela interativa ou, se a
compilação falhar, a janela interativa exibirá todos as mensagens de erro do compilador C#.
O tipo string, que é um alias para a classe System.String, fornece uma série de métodos úteis para pesquisar o
conteúdo de uma cadeia de caracteres. Entre elas estão Contains, StartsWith, EndsWith, IndexOf e LastIndexOf. A
classe System.Text.RegularExpressions.Regex fornece um vocabulário avançado para pesquisar por padrões de
texto. Neste artigo, você aprenderá essas técnicas e como escolher o melhor método para suas necessidades.
string factMessage = "Extension methods have all the capabilities of regular static methods.";
// For user input and strings that will be displayed to the end user,
// use the StringComparison parameter on methods that have it to specify how to match strings.
bool ignoreCaseSearchResult = factMessage.StartsWith("extension",
System.StringComparison.CurrentCultureIgnoreCase);
Console.WriteLine($"Starts with \"extension\"? {ignoreCaseSearchResult} (ignoring case)");
O exemplo anterior demonstra um ponto importante para usar esses métodos. As pesquisas de texto
diferenciam maiúsculas e minúsculas por padrão. Você usa o valor de enumeração
StringComparison.CurrentCultureIgnoreCase para especificar uma pesquisa que não diferencia maiúsculas de
minúsculas.
PADRÃO SIGNIFICADO
string[] sentences =
{
"Put the water over there.",
"They're quite thirsty.",
"Their water bottles broke."
};
if (System.Text.RegularExpressions.Regex.IsMatch(s, sPattern,
System.Text.RegularExpressions.RegexOptions.IgnoreCase))
{
Console.WriteLine($" (match for '{sPattern}' found)");
}
else
{
Console.WriteLine();
}
}
TIP
Os métodos string são geralmente melhores opções quando você está procurando por uma cadeia de caracteres exata.
Expressões regulares são melhores quando você está procurando por algum padrão em uma cadeia de caracteres de
origem.
PADRÃO SIGNIFICADO
if (System.Text.RegularExpressions.Regex.IsMatch(s, sPattern))
{
Console.WriteLine(" - valid");
}
else
{
Console.WriteLine(" - invalid");
}
}
Este padrão de pesquisa único corresponde a várias cadeias de caracteres válidas. Expressões regulares são
melhores para pesquisar por ou validar mediante um padrão, em vez de uma única cadeia de caracteres de texto.
Você pode experimentar estes exemplos examinando o código em nosso repositório GitHub. Ou então, você pode
baixar os exemplos como um arquivo zip.
Consulte também
Guia de Programação em C#
Cadeias de Caracteres
LINQ e Cadeias de Caracteres
System.Text.RegularExpressions.Regex
Expressões regulares do .NET Framework
Linguagem de expressão regular – referência rápida
Práticas recomendadas para o uso de cadeias de caracteres no .NET
Como modificar o conteúdo da cadeia de caracteres
em C#
25/11/2019 • 10 minutes to read • Edit Online
Este artigo demonstra várias técnicas para produzir um string modificando um string existente. Todas as
técnicas demonstradas retornam o resultado das modificações como um novo objeto string . Para demonstrar
isso claramente, todos os exemplos armazenam o resultado em uma nova variável. Em seguida, você pode
examinar o string original e o string resultante da modificação quando você executa cada exemplo.
NOTE
Os exemplos de C# neste artigo são executados no executador de código embutido Try.NET e no playground. Clique no
botão Executar para executar um exemplo em uma janela interativa. Ao executar o código, é possível modificá-lo e executar
o código modificado clicando em Executar novamente. O código modificado será executado na janela interativa ou, se a
compilação falhar, a janela interativa exibirá todos as mensagens de erro do compilador C#.
Há várias técnicas demonstradas neste artigo. Você pode substituir o texto existente. Você pode procurar padrões
e substituir o texto correspondente com outro texto. Você pode tratar uma cadeia de caracteres como uma
sequência de caracteres. Você também pode usar métodos de conveniência que removem o espaço em branco.
Você deve escolher as técnicas que mais se aproximam do seu cenário.
Substituir texto
O código a seguir cria uma nova cadeia de caracteres, substituindo o texto existente por um substituto.
O código anterior demonstra essa propriedade immutable de cadeias de caracteres. Você pode ver no exemplo
anterior que a cadeia de caracteres original, source , não é modificada. O método String.Replace cria uma nova
string contendo as modificações.
O método Replace pode substituir cadeias de caracteres ou caracteres únicos. Em ambos os casos, todas as
ocorrências do texto pesquisado são substituídas. O exemplo a seguir substitui todos os caracteres ' ' com '_':
A cadeia de caracteres de origem não é alterada e uma nova cadeia de caracteres é retornada com a substituição.
Cortar espaço em branco
Você pode usar os métodos String.Trim, String.TrimStart e String.TrimEnd para remover espaços em branco à
esquerda ou à direita. O código a seguir mostra um exemplo de cada um desses casos. A cadeia de caracteres de
origem não é alterada; esses métodos retornam uma nova cadeia de caracteres com o conteúdo modificado.
Remover texto
Você pode remover texto de uma cadeia de caracteres usando o método String.Remove. Esse método remove um
número de caracteres começando em um índice específico. O exemplo a seguir mostra como usar String.IndexOf
seguido por Remove para remover texto de uma cadeia de caracteres:
O método StringBuilder.ToString retorna uma cadeia de caracteres imutável com o conteúdo no objeto
StringBuilder.
string phrase = "The quick brown fox jumps over the fence";
Console.WriteLine(phrase);
unsafe
{
// Compiler will store (intern)
// these strings in same location.
string helloOne = "Hello";
string helloTwo = "Hello";
Você pode experimentar estes exemplos examinando o código em nosso repositório GitHub. Ou então, você pode
baixar os exemplos como um arquivo zip.
Consulte também
Expressões regulares do .NET Framework
Linguagem de expressão regular – referência rápida
Como comparar cadeias de caracteres no C#
25/11/2019 • 18 minutes to read • Edit Online
Você compara cadeias de caracteres para responder uma dessas duas perguntas: "Essas duas cadeias de
caracteres são iguais?" ou "Em que ordem essas cadeias de caracteres devem ser colocadas ao classificá-las?"
Essas duas perguntas são complicadas devido a fatores que afetam as comparações de cadeia de caracteres:
Você pode escolher uma comparação ordinal ou linguística.
Você pode escolher se o uso de maiúsculas faz diferença.
Você pode escolher comparações específicas de cultura.
As comparações linguísticas dependem da plataforma e da cultura.
NOTE
Os exemplos de C# neste artigo são executados no executador de código embutido Try.NET e no playground. Clique no
botão Executar para executar um exemplo em uma janela interativa. Ao executar o código, é possível modificá-lo e executar
o código modificado clicando em Executar novamente. O código modificado será executado na janela interativa ou, se a
compilação falhar, a janela interativa exibirá todos as mensagens de erro do compilador C#.
Ao comparar cadeias de caracteres, você pode definir uma ordem entre elas. As comparações são usadas para
classificar uma sequência de cadeias de caracteres. Com a sequência em uma ordem conhecida, fica mais fácil de
pesquisar, tanto para softwares quanto para os usuários. Outras comparações podem verificar se as cadeias de
caracteres são iguais. Essas verificações são semelhantes a igualdade, mas algumas diferenças, como diferenças
entre maiúsculas e minúsculas, podem ser ignoradas.
Console.WriteLine($"Using == says that <{root}> and <{root2}> are {(root == root2 ? "equal" : "not equal")}");
A comparação ordinal padrão não leva em conta as regras linguísticas ao comparar cadeias de caracteres. Ela
compara o valor binário de cada objeto Char em duas cadeias de caracteres. Como resultado, a comparação
ordinal padrão também diferencia maiúsculas de minúsculas.
Observe que o teste para igualdade com String.Equals e os operadores == e != difere da comparação de
cadeias de caracteres usando os métodos String.CompareTo e Compare(String, String). Embora os testes de
igualdade executem uma comparação ordinal que diferencia maiúsculas de minúsculas, os métodos de
comparação executam uma comparação de diferenciação de maiúsculas e minúsculas, com a cultura atual. Como
os métodos de comparação padrão geralmente fazem diferentes tipos de comparações, recomendamos que você
sempre esclareça a intenção do seu código, chamando uma sobrecarga que especifica explicitamente o tipo de
comparação a ser feita.
Console.WriteLine($"Ordinal ignore case: <{root}> and <{root2}> are {(result ? "equal." : "not equal.")}");
Console.WriteLine($"Ordinal static ignore case: <{root}> and <{root2}> are {(areEqual ? "equal." : "not
equal.")}");
if (comparison < 0)
Console.WriteLine($"<{root}> is less than <{root2}>");
else if (comparison > 0)
Console.WriteLine($"<{root}> is greater than <{root2}>");
else
Console.WriteLine($"<{root}> and <{root2}> are equivalent in order");
Ao fazer uma comparação ordinal sem distinção entre maiúsculas e minúsculas, esses métodos usam as
convenções de maiúsculas e minúsculas da cultura invariável.
Comparações linguísticas
As cadeias de caracteres também podem ser ordenadas usando regras linguísticas para a cultura atual. Às vezes,
isso é conhecido como "ordem de classificação de palavra". Quando você executa uma comparação linguística,
alguns caracteres Unicode não alfanuméricos podem ter pesos especiais atribuídos. Por exemplo, o hífen "-" pode
ter um peso muito pequeno atribuído, de modo que "co-op" e "coop" apareçam próximos um do outro na ordem
de classificação. Além disso, alguns caracteres Unicode podem ser equivalentes a uma sequência de instâncias
Char. O exemplo a seguir usa a frase "They dance in the street." em alemão, com o “ss” (U+0073 U+0073) em
uma cadeia de caracteres e ‘ß’ (U+00DF ) em outra. Linguisticamente (no Windows), "ss" é igual ao caractere 'ß'
Esszet alemão nas culturas "en-US" e "de-DE".
string first = "Sie tanzen auf der Straße.";
string second = "Sie tanzen auf der Strasse.";
showComparison(word, words);
showComparison(word, other);
showComparison(words, other);
void showComparison(string one, string two)
{
int compareLinguistic = String.Compare(one, two, StringComparison.InvariantCulture);
int compareOrdinal = String.Compare(one, two, StringComparison.Ordinal);
if (compareLinguistic < 0)
Console.WriteLine($"<{one}> is less than <{two}> using invariant culture");
else if (compareLinguistic > 0)
Console.WriteLine($"<{one}> is greater than <{two}> using invariant culture");
else
Console.WriteLine($"<{one}> and <{two}> are equivalent in order using invariant culture");
if (compareOrdinal < 0)
Console.WriteLine($"<{one}> is less than <{two}> using ordinal comparison");
else if (compareOrdinal > 0)
Console.WriteLine($"<{one}> is greater than <{two}> using ordinal comparison");
else
Console.WriteLine($"<{one}> and <{two}> are equivalent in order using ordinal comparison");
}
Este exemplo demonstra a natureza dependente de sistema operacional das comparações linguísticas. O host da
janela interativa é um host Linux. As comparações de linguísticas e ordinais produzem os mesmos resultados. Se
você executar este mesmo exemplo em um host do Windows, a saída será a seguinte:
No Windows, a ordem de classificação de "cop", "coop" e "co-op" muda quando você muda de uma comparação
linguística para uma comparação ordinal. As duas frases em alemão também são comparadas de forma diferente
ao usar os tipos diferentes de comparação.
As comparações que diferenciam cultura normalmente são usadas para comparar e classificar cadeias de
caracteres inseridas por usuários com outras cadeias de caracteres inseridas por usuários. Os caracteres e as
convenções de classificação dessas cadeias de caracteres podem variar de acordo com a localidade do
computador do usuário. Até mesmo cadeias de caracteres que contêm caracteres idênticos podem ser
classificadas de formas diferentes dependendo da cultura do thread atual. Além disso, experimente este código de
exemplo localmente em um computador Windows, e você verá os seguintes resultados:
As comparações linguísticas dependem da cultura atual e são dependentes do SO. Você deve considerar isso ao
trabalhar com comparações de cadeia de caracteres.
Classificação linguística e cadeias de caracteres de pesquisa em
matrizes
Os exemplos a seguir mostram como classificar e pesquisar cadeias de caracteres em uma matriz usando uma
comparação linguística que depende da cultura atual. Use os métodos Array estáticos que aceitam um parâmetro
System.StringComparer.
Este exemplo mostra como classificar uma matriz de cadeias de caracteres usando a cultura atual:
Console.WriteLine("Non-sorted order:");
foreach (string s in lines)
{
Console.WriteLine($" {s}");
}
Console.WriteLine("\n\rSorted order:");
Depois que a matriz é classificada, você pode procurar as entradas usando uma pesquisa binária. Uma pesquisa
binária é iniciada no meio da coleção para determinar qual metade da coleção contém a cadeia de caracteres
procurada. Cada comparação subsequente subdivide a parte restante da coleção na metade. A matriz é classificada
usando o StringComparer.CurrentCulture. A função local ShowWhere exibe informações sobre o local em que a
cadeia de caracteres foi encontrada. Se a cadeia de caracteres não for encontrada, o valor retornado indicará o
local em que estaria se fosse encontrada.
string[] lines = new string[]
{
@"c:\public\textfile.txt",
@"c:\public\textFile.TXT",
@"c:\public\Text.txt",
@"c:\public\testfile2.txt"
};
Array.Sort(lines, StringComparer.CurrentCulture);
if (index == 0)
Console.Write("beginning of sequence and ");
else
Console.Write($"{array[index - 1]} and ");
if (index == array.Length)
Console.WriteLine("end of sequence.");
else
Console.WriteLine($"{array[index]}.");
}
else
{
Console.WriteLine($"Found at index {index}.");
}
}
Console.WriteLine("Non-sorted order:");
foreach (string s in lines)
{
Console.WriteLine($" {s}");
}
Console.WriteLine("\n\rSorted order:");
Uma vez classificada, a lista de cadeias de caracteres pode ser pesquisada usando uma pesquisa binária. O
exemplo a seguir mostra como pesquisar a lista classificada usando a mesma função de comparação. A função
local ShowWhere mostra o local em que o texto procurado está ou deveria estar:
List<string> lines = new List<string>
{
@"c:\public\textfile.txt",
@"c:\public\textFile.TXT",
@"c:\public\Text.txt",
@"c:\public\testfile2.txt"
};
lines.Sort((left, right) => left.CompareTo(right));
if (index == 0)
Console.Write("beginning of sequence and ");
else
Console.Write($"{collection[index - 1]} and ");
if (index == collection.Count)
Console.WriteLine("end of sequence.");
else
Console.WriteLine($"{collection[index]}.");
}
else
{
Console.WriteLine($"Found at index {index}.");
}
}
Use sempre o mesmo tipo de comparação para classificação e pesquisa. O uso de tipos diferentes de comparação
para classificação e pesquisa produz resultados inesperados.
Classes de coleção como System.Collections.Hashtable, System.Collections.Generic.Dictionary<TKey,TValue> e
System.Collections.Generic.List<T> têm construtores que usam um parâmetro System.StringComparer quando o
tipo dos elementos ou chaves é string . Em geral, você deve usar esses construtores sempre que possível e
especificar StringComparer.Ordinal ou StringComparer.OrdinalIgnoreCase.
if (String.ReferenceEquals(a, b))
Console.WriteLine("a and b are interned.");
else
Console.WriteLine("a and b are not interned.");
string c = String.Copy(a);
if (String.ReferenceEquals(a, c))
Console.WriteLine("a and c are interned.");
else
Console.WriteLine("a and c are not interned.");
NOTE
Quando você testa cadeias de caracteres quanto a igualdade, é necessário usar os métodos que especificam explicitamente o
tipo de comparação que você pretende executar. O código fica muito mais legível e fácil de manter. Use as sobrecargas dos
métodos das classes System.String e System.Array que aceitam um parâmetro de enumeração StringComparison. Você
especifica o tipo de comparação a ser executado. Evite usar os operadores == e != ao testar a igualdade. Os métodos de
instância String.CompareTo sempre executam uma comparação ordinal que diferencia maiúsculas de minúsculas.
Basicamente, eles são adequados para colocar cadeias de caracteres em ordem alfabética.
Você pode internalizar uma cadeia de caracteres ou recuperar uma referência a uma cadeia de caracteres interna
existente chamando o método String.Intern. Para determinar se uma cadeia de caracteres está internalizada,
chame o método String.IsInterned.
Consulte também
System.Globalization.CultureInfo
System.StringComparer
Cadeias de Caracteres
Comparação de cadeias de caracteres
Globalizando e localizando aplicativos
Como converter com segurança usando
correspondência de padrões e os operadores is e as
25/11/2019 • 7 minutes to read • Edit Online
Como os objetos são polimórficos, é possível que uma variável de tipo de classe base tenha um tipo derivado.
Para acessar os métodos de instância do tipo derivado, é necessário converter o valor de volta no tipo derivado.
No entanto, uma conversão cria o risco de lançar um InvalidCastException. O C# fornece instruções de
correspondência de padrões que executarão uma conversão condicionalmente somente quando ela tiver êxito. O
C# também oferece os operadores is e as para testar se um valor é de um determinado tipo.
O código a seguir demonstra a instrução is da correspondência de padrões. Ele contém métodos que testam um
argumento de método para determinar se ele é um argumento de um possível conjunto de tipos derivados:
class Animal
{
public void Eat() { Console.WriteLine("Eating."); }
public override string ToString()
{
return "I am an animal.";
}
}
class Mammal : Animal { }
class Giraffe : Mammal { }
class SuperNova { }
class Program
{
static void Main(string[] args)
{
Giraffe g = new Giraffe();
FeedMammals(g);
TestForMammals(g);
int? j = null;
PatternMatchingNullable(j);
double d = 9.78654;
PatternMatchingNullable(d);
PatternMatchingSwitch(i);
PatternMatchingSwitch(j);
PatternMatchingSwitch(d);
}
O exemplo anterior demonstra outros recursos da correspondência de padrões a serem usados com conversões.
É possível testar se uma variável tem padrão nulo verificando especificamente o valor null . Quando o valor de
runtime da variável é null , uma instrução is que verifica um tipo retorna sempre false . A instrução is da
correspondência de padrões não permite um tipo de valor anulável, como int? ou Nullable<int> , mas é
possível testar qualquer outro tipo de valor. Os padrões de is do exemplo anterior não estão limitados aos tipos
de valor anulável. Você também pode usar esses padrões para testar se uma variável de um tipo de referência tem
um valor ou é null .
O exemplo anterior também mostra como você usa a expressão is de correspondência de padrões em uma
instrução switch em que a variável pode ser um dos muitos tipos diferentes.
Se desejar testar se uma variável é um determinado tipo, mas não a atribuir à nova variável, será possível usar os
operadores is e as para tipos de referência e tipos que permitem valor nulo. O seguinte código mostra como
usar as instruções is e as que faziam parte da linguagem C# antes que a correspondência de padrões fosse
introduzida para testar se uma variável é de um determinado tipo:
class Animal
{
public void Eat() { Console.WriteLine("Eating."); }
public override string ToString()
{
return "I am an animal.";
}
}
class Mammal : Animal { }
class Giraffe : Mammal { }
class SuperNova { }
class Program
{
static void Main(string[] args)
{
// Use the is operator to verify the type.
// before performing a cast.
Giraffe g = new Giraffe();
UseIsOperator(g);
double d = 9.78654;
UseAsWithNullable(d);
}
Como você pode ver na comparação desse código com o de correspondência de padrões, a sintaxe de
correspondência de padrões oferece recursos mais robustos combinando o teste e a atribuição em uma única
instrução. Use a sintaxe de correspondência de padrões sempre que possível.
Você pode experimentar estes exemplos examinando o código em nosso repositório GitHub. Ou então, você pode
baixar os exemplos como um arquivo zip.
O SDK do .NET Compiler Platform
23/10/2019 • 12 minutes to read • Edit Online
Os compiladores criam um modelo detalhado do código do aplicativo conforme validam a sintaxe e a semântica do
código. O uso desse modelo para criar a saída executável do código-fonte. O SDK do .NET Compiler Platform
fornece acesso a esse modelo. Cada vez mais, contamos com recursos do IDE (ambiente de desenvolvimento
integrado), como IntelliSense, refatoração, renomeação inteligente, "Localizar todas as referências" e "Ir para
definição" para aumentar nossa produtividade. Contamos com ferramentas de análise de código para melhorar a
qualidade e com geradores de código para ajudar na criação do aplicativo. À medida que essas ferramentas ficam
mais inteligentes, elas precisam de acesso a cada vez mais do modelo que somente os compiladores podem criar
conforme processam o código do aplicativo. Este é o objetivo principal das APIs do Roslyn: abrir as caixas pretas e
permitir que as ferramentas e os usuários finais compartilhem a riqueza de informações que os compiladores têm
sobre nosso código. Em vez de serem tradutores opacos, que aceitam código-fonte e produzem código-objeto,
com a Roslyn os compiladores se tornam plataformas: APIs que você pode usar para as tarefas relacionadas ao
código em seus aplicativos e ferramentas.
Próximas etapas
O SDK do .NET Compiler Platform inclui os modelos de objeto de linguagem mais recentes para geração de
código, análise e refatoração. Esta seção fornece uma visão geral conceitual do SDK do .NET Compiler Platform.
Mais detalhes podem ser encontrados nas seções de inícios rápidos, de exemplos e de tutoriais.
Você pode saber mais sobre os conceitos no SDK do .NET Compiler Platform nestes cinco tópicos:
Explorar código com o visualizador de sintaxe
Entender o modelo de API do compilador
Trabalhar com sintaxe
Trabalhar com semântica
Trabalhar com um workspace
Para começar, será necessário instalar o SDK do .NET Compiler Platform:
Os compiladores processam o código escrito seguindo regras estruturadas que geralmente diferem da forma
como os humanos leem e entendem um código. Uma compreensão básica do modelo usado pelos compiladores é
essencial para compreender as APIs usadas ao criar ferramentas baseadas no Roslyn.
Cada fase desse pipeline é um componente separado. Primeiro, a fase de análise cria tokens do texto de origem e
o analisa na sintaxe que segue a gramática da linguagem. Depois, a fase de declaração analisa os metadados de
origem e importados para formar símbolos nomeados. Em seguida, a fase de associação corresponde os
identificadores no código aos símbolos. Por fim, a fase de emissão emite um assembly com todas as informações
criadas pelo compilador.
Correspondente a cada uma dessas fases, o SDK do .NET Compiler Platform expõe um modelo de objeto que
permite o acesso às informações da fase. A fase de análise expõe uma árvore de sintaxe, a fase de declaração
expõe uma tabela de símbolos hierárquica, a fase de associação expõe o resultado da análise semântica do
compilador e a fase de emissão é uma API que gera códigos de bytes de IL.
Cada compilador combina esses componentes como um único inteiro de ponta a ponta.
Essas APIs são as mesmas usadas pelo Visual Studio. Por exemplo, os recursos de formatação e estrutura de
tópicos do código usam as árvores de sintaxe, o Pesquisador de Objetos e os recursos de navegação usam a tabela
de símbolos, as refatorações e o recurso Ir para Definição usam o modelo semântico e o recurso Editar e
Continuar usa todos eles, incluindo a API de Emissão.
Camadas de API
O SDK do .NET Compiler consiste em duas camadas principais de APIs: APIs do compilador e APIs dos
workspaces.
APIs do compilador
A camada do compilador contém os modelos de objeto que correspondem às informações expostas em cada fase
do pipeline do compilador, tanto sintáticas quanto semânticas. A camada do compilador também contém um
instantâneo imutável de uma única invocação de um compilador, incluindo referências de assembly, opções do
compilador e arquivos de código-fonte. Há duas APIs distintas que representam a linguagem C# e a linguagem
Visual Basic. Essas duas APIs são semelhantes na forma, mas adaptadas para alta fidelidade a cada linguagem
individual. Essa camada não tem dependências em componentes do Visual Studio.
APIs de diagnóstico
Como parte de sua análise, o compilador pode produzir um conjunto de diagnósticos que abrangem tudo, desde a
sintaxe, semântica e erros de atribuição definitiva a vários diagnósticos de avisos e informativos. A camada de API
do Compilador expõe o diagnóstico por meio de uma API extensível que permite que os analisadores definidos
pelo usuário sejam conectados ao processo de compilação. Ela possibilita que o diagnóstico definido pelo usuário,
como aqueles gerados por ferramentas como o StyleCop ou FxCop, seja produzido junto com o diagnóstico
definido pelo compilador. A produção de diagnóstico dessa maneira tem o benefício da integração natural a
ferramentas como o MSBuild e Visual Studio, que dependem do diagnóstico para experiências como interrupção
de um build com base na política, exibição de rabiscos em tempo real no editor e sugestão de correções de código.
APIs de script
APIs de hospedagem e script fazem parte da camada do compilador. Você pode usá-las para execução de snippets
de código e acúmulo de um contexto de execução em tempo de execução. O REPL (Loop de Leitura-Avaliação-
Impressão) interativo do C# usa essas APIs. O REPL permite usar o C# como a linguagem de scripts, executando
o código de forma interativa à medida que ele é escrito.
APIs dos workspaces
A camada Workspaces contém a API de Workspace, que é o ponto de partida para fazer a análise de código e
refatoração em soluções inteiras. Ela ajuda você a organizar todas as informações sobre os projetos de uma
solução em um único modelo de objeto, oferecendo acesso direto aos modelos de objeto da camada do
compilador, sem a necessidade de analisar arquivos, configurar opções ou gerenciar dependências de projeto a
projeto.
Além disso, a camada Workspaces expõe um conjunto de APIs usado ao implementar ferramentas de análise de
código e refatoração que funcionam em um ambiente de host como o IDE do Visual Studio. Exemplos incluem as
APIs Localizar Todas as Referências, Formatação e Geração de Código.
Essa camada não tem dependências em componentes do Visual Studio.
Trabalhar com sintaxe
24/10/2019 • 16 minutes to read • Edit Online
A árvore de sintaxe é uma estrutura de dados fundamental exposta pelas APIs do compilador. Essas árvores
representam a estrutura lexical e sintática do código-fonte. Elas servem duas finalidades importantes:
1. Permitir que ferramentas – como um IDE, suplementos, ferramentas de análise de código e refatorações –
vejam e processem a estrutura sintática do código-fonte no projeto do usuário.
2. Permitir que ferramentas – como refatorações e um IDE – criem, modifiquem e reorganizem o código-fonte de
uma maneira natural sem a necessidade de uso de edições de texto diretas. Criando e manipulando árvores, as
ferramentas podem criar e reorganizar o código-fonte com facilidade.
Árvores de sintaxe
Árvores de sintaxe são a estrutura principal usada para compilação, análise de código, associação, refatoração,
recursos de IDE e geração de código. Nenhuma parte do código-fonte é entendida sem primeiro ser identificada e
categorizada em um dos muitos elementos de linguagem estrutural conhecidos.
As árvores de sintaxe têm três atributos-chave. O primeiro atributo é que as árvores de sintaxe mantêm todas as
informações de origem em fidelidade total. Isso significa que a árvore de sintaxe contém cada informação
encontrada no texto de origem, cada constructo gramatical, cada token lexical e todo o resto, incluindo espaço em
branco, comentários e diretivas do pré-processador. Por exemplo, cada literal mencionado na fonte é representado
exatamente como foi digitado. As árvores de sintaxe também representam erros no código-fonte quando o
programa está incompleto ou mal-formado, com a representação de tokens ignorados ou ausentes na árvore de
sintaxe.
Isso possibilita o segundo atributo das árvores de sintaxe. Uma árvore de sintaxe obtida do analisador pode
produzir o texto exato com base no qual ela foi analisada. Em qualquer nó de sintaxe, é possível obter a
representação de texto da subárvore com raiz nesse nó. Isso significa que as árvores de sintaxe podem ser usadas
como uma maneira de construir e editar o texto de origem. Ao criar uma árvore, por implicação, você criou o texto
equivalente e, ao editar uma árvore de sintaxe criando uma nova árvore com base nas alterações de uma árvore
existente, você editou o texto efetivamente.
O terceiro atributo das árvores de sintaxe é que elas são imutáveis e thread-safe. Isso significa que, depois que
uma árvore é obtida, ela é um instantâneo do estado atual do código e nunca é alterada. Isso permite que vários
usuários interajam com a mesma árvore de sintaxe ao mesmo tempo em threads diferentes sem bloqueio nem
duplicação. Como as árvores são imutáveis e nenhuma modificação pode ser feita diretamente em uma árvore, os
métodos de fábrica ajudam a criar e modificar árvores de sintaxe criando instantâneos adicionais da árvore. As
árvores são eficientes no modo como reutilizam os nós subjacentes, de forma que uma nova versão possa ser
recompilada rapidamente e com pouca memória extra.
Uma árvore de sintaxe é literalmente uma estrutura de dados de árvore, em que os elementos estruturais não
terminais são pais de outros elementos. Cada árvore de sintaxe é composta por nós, tokens e desafios.
Nós de sintaxe
Nós de sintaxe são um dos elementos principais das árvores de sintaxe. Esses nós representam os constructos
sintáticos como declarações, instruções, cláusulas e expressões. Cada categoria de nós de sintaxe é representada
por uma classe separada derivada de Microsoft.CodeAnalysis.SyntaxNode. O conjunto de classes de nó não é
extensível.
Todos os nós de sintaxe são nós não terminais na árvore de sintaxe, o que significa que eles sempre têm outros
nós e tokens como filhos. Como filho de outro nó, cada nó tem um nó pai que pode ser acessado por meio da
propriedade SyntaxNode.Parent. Como os nós e as árvores são imutáveis, o pai de um nó nunca é alterado. A raiz
da árvore tem um pai nulo.
Cada nó tem um método SyntaxNode.ChildNodes(), que retorna uma lista de nós filho em ordem sequencial com
base em sua posição no texto de origem. Essa lista não contém tokens. Cada nó também tem métodos para
examinar descendentes, como DescendantNodes, DescendantTokens ou DescendantTrivia-que representam uma
lista de todos os nós, tokens ou Trívia que existem na subárvore com raiz por esse nó.
Além disso, cada subclasse de nó de sintaxe expõe os mesmos filhos por meio de propriedades fortemente
tipadas. Por exemplo, uma classe de nó BinaryExpressionSyntax tem três propriedades adicionais específicas aos
operadores binários: Left, OperatorToken e Right. O tipo de Left e Right é ExpressionSyntax e o tipo de
OperatorToken é SyntaxToken.
Alguns nós de sintaxe têm filhos opcionais. Por exemplo, um IfStatementSyntax tem um ElseClauseSyntax
opcional. Se o filho não estiver presente, a propriedade retornará nulo.
Tokens de sintaxe
Os tokens de sintaxe são os terminais da gramática da linguagem, que representam os menores fragmentos
sintáticos do código. Eles nunca são os pais de outros nós ou tokens. Os tokens de sintaxe consistem em palavras-
chave, identificadores, literais e pontuação.
Para fins de eficiência, o tipo SyntaxToken é um tipo de valor CLR. Portanto, ao contrário dos nós de sintaxe, há
apenas uma estrutura para todos os tipos de tokens com uma combinação de propriedades que têm significado,
dependendo do tipo de token que está sendo representado.
Por exemplo, um token literal inteiro representa um valor numérico. Além do texto de origem não processado
abrangido pelo token, o token literal tem uma propriedade Value que informa o valor inteiro decodificado exato.
Essa propriedade é tipada como Object porque pode ser um dos muitos tipos primitivos.
A propriedade ValueText indica as mesmas informações que a propriedade Value; no entanto, essa propriedade
sempre é tipada como String. Um identificador no texto de origem C# pode incluir caracteres de escape Unicode,
embora a sintaxe da sequência de escape em si não seja considerada parte do nome do identificador. Portanto,
embora o texto não processado abrangido pelo token inclua a sequência de escape, isso não ocorre com a
propriedade ValueText. Em vez disso, ela inclui os caracteres Unicode identificados pelo escape. Por exemplo, se o
texto de origem contiver um identificador gravado como \u03C0 , a propriedade ValueText desse token retornará
π .
Desafios de sintaxe
Os desafios de sintaxe representam as partes do texto de origem que são amplamente insignificantes para o
reconhecimento normal do código, como espaço em branco, comentários e diretivas do pré-processador. Assim
como os tokens de sintaxe, os desafios são tipos de valor. O único tipo Microsoft.CodeAnalysis.SyntaxTrivia é
usado para descrever todos os tipos de desafios.
Como os desafios não fazem parte da sintaxe de linguagem normal e podem aparecer em qualquer lugar entre
dois tokens quaisquer, eles não são incluídos na árvore de sintaxe como um filho de um nó. Apesar disso, como
eles são importantes ao implementar um recurso como refatoração e para manter fidelidade total com o texto de
origem, eles existem como parte da árvore de sintaxe.
Acesse os desafios inspecionando as coleções SyntaxToken.LeadingTrivia ou SyntaxToken.TrailingTrivia de um
token. Quando o texto de origem é analisado, sequências de desafios são associadas aos tokens. Em geral, um
token possui qualquer desafio após ele na mesma linha até o próximo token. Qualquer desafio após essa linha é
associado ao próximo token. O primeiro token no arquivo de origem obtém todos as desafios iniciais e a última
sequência de desafios no arquivo é anexada ao token de fim do arquivo, que, de outro modo, tem largura zero.
Ao contrário dos nós e tokens de sintaxe, os desafios de sintaxe não têm pais. Apesar disso, como eles fazem parte
da árvore e cada um deles é associado um único token, você poderá acessar o token ao qual ele está associado
usando a propriedade SyntaxTrivia.Token.
Intervalos
Cada nó, token ou desafio conhece sua posição dentro do texto de origem e o número de caracteres no qual ele
consiste. Uma posição de texto é representada como um inteiro de 32 bits, que é um índice char baseado em
zero. Um objeto TextSpan é a posição inicial e uma contagem de caracteres, ambas representadas como inteiros.
Se TextSpan tem comprimento zero, ele se refere a um local entre dois caracteres.
Cada nó tem duas propriedades TextSpan: Span e FullSpan.
A propriedade Span é o intervalo de texto do início do primeiro token na subárvore do nó ao final do último
token. Esse intervalo não inclui nenhum desafio à esquerda ou à direita.
A propriedade FullSpan é o intervalo de texto que inclui o intervalo normal do nó mais o intervalo de qualquer
desafio à esquerda ou à direita.
Por exemplo:
if (x > 3)
{
|| // this is bad
|throw new Exception("Not right.");| // better exception?||
}
O nó de instrução dentro do bloco tem um intervalo indicado pelas barras verticais simples (|). Ele inclui os
caracteres throw new Exception("Not right."); . O intervalo total é indicado pelas barras verticais duplas (||). Ele
inclui os mesmos caracteres do intervalo e os caracteres associados ao desafio à esquerda e à direita.
Variantes
Cada nó, token ou desafio tem uma propriedade SyntaxNode.RawKind, do tipo System.Int32, que identifica o
elemento de sintaxe exato representado. Esse valor pode ser convertido em uma enumeração específica a uma
linguagem; cada linguagem, C# ou VB, tem uma única enumeração SyntaxKind
(Microsoft.CodeAnalysis.CSharp.SyntaxKind e Microsoft.CodeAnalysis.VisualBasic.SyntaxKind, respectivamente)
que lista todos os possíveis elementos de nós, tokens e desafios na gramática. Esta conversão pode ser feita
automaticamente acessando os métodos de extensão CSharpExtensions.Kind ou VisualBasicExtensions.Kind.
A propriedade RawKind permite a desambiguidade fácil de tipos de nó de sintaxe que compartilham a mesma
classe de nó. Para tokens e desafios, essa propriedade é a única maneira de diferenciar um tipo de elemento de
outro.
Por exemplo, uma única classe BinaryExpressionSyntax tem Left, OperatorToken e Right como filhos. A
propriedade Kind distingue se ela é um tipo AddExpression, SubtractExpression ou MultiplyExpression de nó de
sintaxe.
Erros
Mesmo quando o texto de origem contém erros de sintaxe, uma árvore de sintaxe completa com ida e volta para a
origem é exposta. Quando o analisador encontra um código que não está em conformidade com a sintaxe
definida da linguagem, ele usa uma das duas técnicas para criar uma árvore de sintaxe.
Primeiro, se o analisador espera determinado tipo de token, mas não o encontra, ele pode inserir um token
ausente na árvore de sintaxe no local em que o token era esperado. Um token ausente representa o token real que
era esperado, mas tem um intervalo vazio e sua propriedade SyntaxNode.IsMissing retorna true .
Em segundo lugar, o analisador pode ignorar tokens até encontrar um no qual pode continuar a análise. Nesse
caso, os tokens ignorados são anexados como um nó de desafio com o tipo SkippedTokensTrivia.
Trabalhar com semântica
23/10/2019 • 6 minutes to read • Edit Online
As árvores de sintaxe representam a estrutura lexical e sintática do código-fonte. Embora essas informações
apenas sejam suficientes para descrever todas as declarações e a lógica na fonte, não são informações suficientes
para identificar o que está sendo referenciado. Um nome pode representar:
um tipo
um campo
um método
uma variável local
Embora cada um deles seja exclusivamente diferente, determinar a qual deles um identificador, de fato, se refere
exige uma compreensão profunda das regras da linguagem.
Há elementos do programa representados no código-fonte e os programas também podem se referir a
bibliotecas compiladas anteriormente, empacotadas em arquivos do assembly. Embora nenhum código-fonte e,
portanto, nenhum nó ou árvore de sintaxe, esteja disponível para assemblies, os programas ainda podem se
referir a elementos contidos neles.
Para essas tarefas, é necessário usar o Modelo semântico.
Além de um modelo sintático do código-fonte, um modelo semântico encapsula as regras da linguagem,
fornecendo uma maneira fácil para fazer a correspondência correta de identificadores com o elemento de
programa correto que está sendo referenciado.
Compilação
Uma compilação é uma representação de tudo o que é necessário para compilar um programa do C# ou Visual
Basic, que inclui todas as referências de assembly, opções do compilador e arquivos de origem.
Como todas essas informações estão em um só lugar, os elementos contidos no código-fonte podem ser descritos
mais detalhadamente. A compilação representa cada tipo, membro ou variável declarada como um símbolo. A
compilação contém uma variedade de métodos que ajudam você encontrar e identificar os símbolos que foram
declarados no código-fonte ou importados como metadados de um assembly.
Semelhantes às árvores de sintaxe, as compilações são imutáveis. Depois de criar uma compilação, ela não pode
ser alterada por você ou por outra pessoa com quem você pode está compartilhando. No entanto, é possível criar
uma nova compilação com base em uma compilação existente, especificando uma alteração conforme ela é feita.
Por exemplo, é possível criar uma compilação que é igual em todos os aspectos a uma compilação existente, com
exceção de que ela pode incluir um arquivo de origem ou uma referência de assembly adicional.
Símbolos
Um símbolo representa um elemento distinto declarado pelo código-fonte ou importado de um assembly como
metadados. Cada namespace, tipo, método, propriedade, campo, evento, parâmetro ou variável local é
representado por um símbolo.
Uma variedade de métodos e propriedades no tipo Compilation ajudam você a encontrar símbolos. Por exemplo,
encontre um símbolo para um tipo declarado pelo seu nome comum de metadados. Também acesse a tabela
inteira de símbolos como uma árvore de símbolos com raiz no namespace global.
Os símbolos também contêm informações adicionais que o compilador determina da fonte ou dos metadados,
como outros símbolos referenciados. Cada tipo de símbolo é representado por uma interface separada derivada
de ISymbol, cada um com seus próprios métodos e propriedades que fornecem detalhes das informações
reunidas pelo compilador. Muitas dessas propriedades referenciam outros símbolos diretamente. Por exemplo, a
propriedade IMethodSymbol.ReturnType informa o símbolo de tipo real referenciado pela declaração de método.
Os símbolos apresentam uma representação comum de namespaces, tipos e membros, entre o código-fonte e os
metadados. Por exemplo, um método que foi declarado no código-fonte e um método que foi importado dos
metadados são representados por um IMethodSymbol com as mesmas propriedades.
Símbolos são semelhantes em conceito ao sistema de tipos CLR, conforme representado pela API
System.Reflection, mas são mais sofisticados pois modelam mais do que apenas tipos. Namespaces, variáveis
locais e rótulos são todos símbolos. Além disso, os símbolos são uma representação dos conceitos da linguagem,
não dos conceitos do CLR. Há muita sobreposição, mas há muitas diferenças significativas também. Por exemplo,
um método iterador no C# ou Visual Basic é um único símbolo. No entanto, quando o método iterador é
convertido em metadados CLR, ele é um tipo e vários métodos.
Modelo semântico
Um modelo semântico representa todas as informações semânticas de um único arquivo de origem. Use-o para
descobrir o seguinte:
Os símbolos referenciados em um local específico na fonte.
O tipo resultante de qualquer expressão.
Todo o diagnóstico, que são erros e avisos.
Como as variáveis fluem bidirecionalmente entre as regiões de origem.
As respostas a perguntas mais especulativas.
Trabalhar com um espaço de trabalho
22/06/2018 • 5 minutes to read • Edit Online
A camada Espaços de Trabalho contém a API de Espaço de Trabalho para fazer a análise de código e refatoração
em soluções inteiras. Nessa camada, a API do Espaço de Trabalho ajudará você a organizar todas as informações
sobre os projetos de uma solução em um único modelo de objeto, oferecendo acesso direto aos modelos de objeto
da camada do compilador como texto de origem, árvores de sintaxe, modelos semânticos e compilações, sem a
necessidade de analisar arquivos, configurar opções ou gerenciar dependências entre projetos.
Ambientes de host, como um IDE, fornecem um espaço de trabalho para você correspondente à solução aberta.
Também é possível usar esse modelo fora de um IDE apenas carregando um arquivo de solução.
Espaço de trabalho
Um espaço de trabalho é uma representação ativa da solução como uma coleção de projetos, cada uma com uma
coleção de documentos. Um espaço de trabalho normalmente é vinculado a um ambiente de host que está sendo
modificado constantemente conforme um usuário digita ou manipula propriedades.
O Workspace fornece acesso ao modelo atual da solução. Quando ocorre uma alteração no ambiente de host, o
espaço de trabalho aciona eventos correspondentes e a propriedade Workspace.CurrentSolution é atualizada. Por
exemplo, quando o usuário digita em um editor de texto algo correspondente a um dos documentos de origem, o
espaço de trabalho usa um evento para sinalizar que o modelo geral da solução foi alterado e qual documento foi
modificado. Em seguida, você pode reagir a essas alterações analisando o novo modelo quanto à exatidão,
realçando áreas de significância ou fazendo uma sugestão de alteração de código.
Também pode criar espaços de trabalho independentes desconectados do ambiente de host ou usados em um
aplicativo que não tem nenhum ambiente de host.
Este artigo oferece uma visão geral sobre a ferramenta de Visualizador de sintaxe que é fornecida como parte do
SDK do .NET Compiler Platform ("Roslyn"). O Visualizador de sintaxe é uma janela de ferramentas que ajuda a
inspecionar e explorar árvores de sintaxe. É uma ferramenta essencial para compreender os modelos do código
que você deseja analisar. Também é um auxílio para depuração ao desenvolver seus próprios aplicativos usando o
SDK do.NET Compiler Platform ("Roslyn"). Abra essa ferramenta ao criar seus primeiros analisadores. O
visualizador ajuda você a entender os modelos usados pelas APIs. Você também pode usar ferramentas como
SharpLab ou LINQPad para inspecionar o código e entender as árvores de sintaxe.
Visualizador de sintaxe
O Visualizador de sintaxe permite a inspeção da árvore de sintaxe de arquivo de código C# ou VB na janela do
editor atualmente ativa dentro do IDE do Visual Studio. O visualizador pode ser iniciado ao clicar em Exibir >
Outras Janelas > Visualizador de sintaxe. Você também pode usar a barra de ferramentas de Início Rápido no
canto superior direito. Digite "sintaxe" e o comando para abrir o Visualizador de sintaxe deverá aparecer.
Este comando abre o Visualizador de sintaxe como uma janela de ferramentas flutuante. Se você não tiver uma
janela de editor de código aberta, a exibição ficará em branco, conforme mostrado na figura a seguir.
Encaixe esta janela de ferramentas em um local conveniente dentro do Visual Studio, como o lado esquerdo. O
Visualizador mostra informações sobre o arquivo de código atual.
Crie um novo projeto usando o comando Arquivo > Novo Projeto. Você pode criar um projeto do VB ou C#.
Quando o Visual Studio abre o arquivo de código principal deste projeto, o visualizador exibe a árvore de sintaxe
dele. Você pode abrir qualquer arquivo de C# ou VB existente nesta instância do Visual Studio e o visualizador
exibirá a árvore de sintaxe do arquivo correspondente. Se você tiver vários arquivos de código abertos no Visual
Studio, o visualizador exibirá a árvore de sintaxe do arquivo de código atualmente ativo, (o arquivo de código que
tem o foco do teclado).
C#
Visual Basic
Conforme mostrado nas imagens acima, a janela de ferramentas do visualizador exibe a árvore de sintaxe na parte
superior e uma grade de propriedade na parte inferior. A grade de propriedade exibe as propriedades do item
atualmente selecionado na árvore, incluindo Type e Kind (SyntaxKind) do .NET do item.
As árvores de sintaxe incluem três tipos de itens: nós, tokens e trívia. Você pode ler mais sobre esses tipos no
artigo Trabalhar com sintaxe. Os itens de cada tipo estão representados com cores diferentes. Clique no botão
"Legenda" para uma visão geral das cores usadas.
Cada item da árvore também exibe sua própria extensão. A extensão é composta pelos índices (a posição inicial
e final) do nó no arquivo de texto. No exemplo de C# anterior, o token “UsingKeyword [0..5)” selecionado tem uma
abrangência de cinco caracteres de largura, [0..5). A notação "[..)" significa que o índice inicial faz parte da
extensão, mas o índice final não.
Há duas maneiras de navegar na árvore:
Expandir ou clicar em itens na árvore. O visualizador seleciona automaticamente o texto correspondente à
extensão do item no editor de código.
Clicar ou selecionar texto no editor de código. No exemplo anterior do VB, se você seleciona a linha que contém
"Module Module1" no editor de código, o visualizador navega automaticamente até o nó ModuleStatement
correspondente na árvore.
O visualizador realça o item da árvore cuja extensão melhor corresponda com a extensão do texto selecionado no
editor.
O visualizador atualiza a árvore para corresponder às modificações no arquivo de código ativo. Adicione uma
chamada a Console.WriteLine() dentro de Main() . Conforme você digita, o visualizador atualiza a árvore.
Pare de digitar quando já tiver digitado Console. . A árvore tem alguns itens coloridos em rosa. Neste ponto, há
erros (também conhecidos como "Diagnóstico") no código digitado. Esses erros são anexados a nós, tokens e trívia
na árvore de sintaxe. O visualizador mostra quais itens têm erros anexados a eles, realçando a tela de fundo em
rosa. Você pode inspecionar os erros em qualquer item colorido em rosa passando o mouse sobre o item. O
visualizador exibe somente erros sintáticos (os erros relacionados à sintaxe do código digitado); erros semânticos
não são exibidos.
Gráficos de sintaxe
Clique com o botão direito do mouse em qualquer item da árvore e clique em Exibir gráfico de sintaxe
direcionado.
C#
Visual Basic
O visualizador exibe uma representação gráfica da subárvore com raiz no item selecionado. Repita essas etapas
para o nó MethodDeclaration correspondente ao método Main() no exemplo de C#. O visualizador exibe um
gráfico de sintaxe que tem a seguinte aparência:
O visualizador de gráfico de sintaxe tem uma opção para exibir uma legenda de seu esquema de cores. Você
também pode passar o mouse sobre itens específicos no gráfico de sintaxe para exibir as respectivas propriedades.
Você pode exibir gráficos de sintaxe de itens diferentes da árvore repetidamente e os gráficos serão sempre
exibidos na mesma janela no Visual Studio. Você pode encaixar essa janela em um local conveniente no Visual
Studio para não precisar alternar entre as guias para exibir um novo gráfico de sintaxe. A parte inferior, abaixo das
janelas do editor de código, geralmente é conveniente.
Veja o layout de encaixe para usar com a janela de ferramentas do visualizador e a janela do gráfico de sintaxe:
Outra opção é colocar a janela de gráfico de sintaxe em um segundo monitor, em uma configuração de dois
monitores.
Inspecionando semântica
O Visualizador de sintaxe possibilita uma inspeção rudimentar de símbolos e informações semânticas. Digite
double x = 1 + 1; dentro de Main() no exemplo de C#. Em seguida, selecione a expressão 1 + 1 na janela do
editor de código. O visualizador realça o nó AddExpression no visualizador. Clique com o botão direito do mouse
nesse AddExpression e clique em Exibir Symbol (se houver) . Observe que a maioria dos itens de menu tem o
qualificador "se houver". O Visualizador de sintaxe inspeciona as propriedades de um Nó, incluindo propriedades
que podem não estar presentes em todos os nós.
A grade de propriedades nas atualizações do visualizador, como mostra a figura a seguir: O símbolo da expressão
é um SynthesizedIntrinsicOperatorSymbol com Kind = Method.
Experimente Exibir TypeSymbol (se houver) para o mesmo nó AddExpression. A grade de propriedade no
visualizador é atualizada, conforme mostrado na figura a seguir, indicando que o tipo da expressão selecionada é
Int32 .
Experimente Exibir TypeSymbol Convertido (se houver) para o mesmo nó AddExpression. A grade de
propriedade é atualizada, indicando que, embora o tipo da expressão seja Int32 , o tipo convertido da expressão é
Double , conforme mostrado na figura a seguir. Esse nó inclui informações de símbolo de tipo convertido porque a
expressão Int32 ocorre em um contexto em que deve ser convertida em um Double . Essa conversão satisfaz o
tipo Double especificado para a variável x no lado esquerdo do operador de atribuição.
Por fim, experimente Exibir Valor Constante (se houver) para o mesmo nó AddExpression. A grade de
propriedade mostra que o valor da expressão é uma constante de tempo de compilação com o valor 2 .
O exemplo anterior também pode ser replicado no VB. Digite Dim x As Double = 1 + 1 em um arquivo do VB.
Selecione a expressão 1 + 1 na janela do editor de código. O visualizador realça o nó AddExpression
correspondente no visualizador. Repita as etapas anteriores para esta AddExpression e você verá resultados
idênticos.
Examine mais código no VB. Atualize seu arquivo principal do VB com o seguinte código:
Imports C = System.Console
Module Program
Sub Main(args As String())
C.WriteLine()
End Sub
End Module
Esse código introduz um alias chamado C que mapeia para o tipo System.Console na parte superior do arquivo e
usa esse alias em Main() . Selecione o uso desse alias, o C em C.WriteLine() , dentro do método Main() . O
visualizador seleciona o nó IdentifierName correspondente no visualizador. Clique com botão direito do mouse
nesse nó e clique em Exibir Symbol (se houver) . A grade de propriedade mostra que esse identificador é
associado com o tipo System.Console , conforme mostrado na figura a seguir:
Experimente Exibir AliasSymbol (se houver) para o mesmo nó IdentifierName. A grade de propriedade
mostra que o identificador é um alias com nome C , que é associado com o destino System.Console . Em outras
palavras, a grade de propriedade fornece informações sobre o AliasSymbol correspondente ao identificador C .
Inspecione o símbolo correspondente a qualquer tipo, método e propriedade declarados. Selecione o nó
correspondente no visualizador e clique em Exibir Symbol (se houver) . Selecione o método Sub Main() ,
incluindo o corpo do método. Clique em Exibir Symbol (se houver) para o nó SubBlock correspondente no
visualizador. A grade de propriedade mostra que o MethodSymbol deste SubBlock tem nome Main com o tipo
de retorno Void .
Os exemplos de VB acima podem ser facilmente replicados em C#. Digite using C = System.Console; no lugar de
Imports C = System.Console para o alias. As etapas anteriores em C# geram resultados idênticos na janela do
visualizador.
As operações de inspeção semântica estão disponíveis somente em nós. Elas não estão disponíveis em tokens nem
trívia. Nem todos os nós têm informações semânticas interessantes para inspecionar. Quando um nó não tem
informações semânticas interessantes, clicar em Exibir * Symbol (se houver) mostrará uma grade de
propriedade em branco.
Você pode ler mais sobre as APIs para executar análise semântica no documento de visão geral Trabalhar com
semântica.
Neste tutorial, você explorará a API de sintaxe. A API de sintaxe fornece acesso às estruturas de dados que
descrevem um programa C# ou Visual Basic. Essas estruturas de dados têm detalhes suficientes para que possam
representar qualquer programa, de qualquer tamanho. Essas estruturas podem descrever programas completos
que compilam e executam corretamente. Elas também podem descrever programas incompletos, enquanto você
os escreve no editor.
Para habilitar essa expressão avançada, as estruturas de dados e as APIs que compõem a API de sintaxe são
necessariamente complexas. Começaremos com a aparência da estrutura de dados para o programa "Olá,
Mundo" típico:
using System;
using System.Collections.Generic;
using System.Linq;
namespace HelloWorld
{
class Program
{
static void Main(string[] args)
{
Console.WriteLine("Hello World!");
}
}
}
Veja o texto do programa anterior. Você reconhece elementos familiares. O texto inteiro representa um único
arquivo de origem, ou uma unidade de compilação. As três primeiras linhas do arquivo de origem são
diretivas de uso. A origem restante está contida em uma declaração de namespace. A declaração de
namespace contém uma declaração de classe filha. A declaração de classe contém uma declaração de método.
A API de sintaxe cria uma estrutura de árvore com a raiz que representa a unidade de compilação. Nós da árvore
representam as diretivas using, a declaração de namespace e todos os outros elementos do programa. A estrutura
de árvore continua até os níveis mais baixos: a cadeia de caracteres "Olá, Mundo!" é um token literal da cadeia
de caracteres que é um descendente de um argumento. A API de sintaxe fornece acesso à estrutura do
programa. Você pode consultar as práticas recomendadas de código específico, percorrer a árvore inteira para
entender o código e criar novas árvores ao modificar a árvore existente.
Essa breve descrição fornece uma visão geral do tipo de informações acessíveis usando a API de sintaxe. A API de
sintaxe não é nada mais de uma API formal que descreve os constructos de código familiares que você conhece
do C#. As funcionalidades completas incluem informações sobre como o código é formatado, incluindo quebras
de linha, espaço em branco e recuo. Usando essas informações, você pode representar totalmente o código como
escrito e lido por programadores humanos ou pelo compilador. Usar essa estrutura permite que você interaja com
o código-fonte em um nível muito significativo. Não se trata mais de cadeias de caracteres de texto, mas de dados
que representam a estrutura de um programa C#.
Para começar, será necessário instalar o SDK do .NET Compiler Platform:
Ao navegar nessa estrutura de árvore, você pode encontrar qualquer instrução, expressão, token ou bit de espaço
em branco em um arquivo de código.
Embora você possa encontrar tudo em um arquivo de código usando as APIs de sintaxe, a maioria dos cenários
envolvem o exame de pequenos trechos de código ou a pesquisa por instruções ou fragmentos específicos. Os
dois exemplos a seguir mostram usos típicos para navegar pela estrutura de códigos ou pesquisar por instruções
individuais.
Percorrendo árvores
Você pode examinar os nós em uma árvore de sintaxe de duas maneiras. Você pode percorrer a árvore para
examinar cada nó, ou então consultar elementos ou nós específicos.
Passagem manual
Você pode ver o código concluído para essa amostra no nosso repositório do GitHub.
NOTE
Os tipos de árvore de sintaxe usam a herança para descrever os elementos de sintaxe diferentes que são válidos em locais
diferentes no programa. Usar essas APIs geralmente significa converter propriedades ou membros da coleção em tipos
derivados específicos. Nos exemplos a seguir, a atribuição e as conversões são instruções separadas, usando variáveis
explicitamente tipadas. Você pode ler o código para ver os tipos de retorno da API e o tipo de runtime dos objetos
retornados. Na prática, é mais comum usar variáveis implicitamente tipadas e depender de nomes de API para descrever o
tipo de objeto que está sendo examinado.
namespace HelloWorld
{
class Program
{
static void Main(string[] args)
{
Console.WriteLine(""Hello, World!"");
}
}
}";
Em seguida, adicione o código a seguir para criar a árvore de sintaxe para o texto do código na constante
programText . Adicione a seguinte linha ao seu método Main :
Essas duas linhas criam a árvore e recuperam o nó raiz dessa árvore. Agora você pode examinar os nós na árvore.
Adicione essas linhas ao seu método Main para exibir algumas das propriedades do nó raiz na árvore:
Execute o aplicativo para ver o que seu código descobriu sobre o nó raiz nessa árvore.
Normalmente, percorreria a árvore para saber mais sobre o código. Neste exemplo, você está analisando código
que você conhece para explorar as APIs. Adicione o código a seguir para examinar o primeiro membro do nó
root :
O nó de declaração do método contém todas as informações de sintaxe sobre o método. Permite exibir o tipo de
retorno do método Main , o número e os tipos dos argumentos e o texto do corpo do método. Adicione o seguinte
código:
Execute o programa para ver todas as informações que você descobriu sobre este programa:
The tree is a CompilationUnit node.
The tree has 1 elements in it.
The tree has 4 using statements. They are:
System
System.Collections
System.Linq
System.Text
The first member is a NamespaceDeclaration.
There are 1 members declared in this namespace.
The first member is a ClassDeclaration.
There are 1 members declared in the Program class.
The first member is a MethodDeclaration.
The return type of the Main method is void.
The method has 1 parameters.
The type of the args parameter is string[].
The body text of the Main method follows:
{
Console.WriteLine("Hello, World!");
}
Métodos de consulta
Além de percorrer árvores, você também pode explorar a árvore de sintaxe usando os métodos de consulta
definidos em Microsoft.CodeAnalysis.SyntaxNode. Esses métodos devem ser imediatamente familiares a qualquer
pessoa familiarizada com o XPath. Você pode usar esses métodos com o LINQ para localizar itens rapidamente
em uma árvore. O SyntaxNode tem métodos de consulta como DescendantNodes, AncestorsAndSelf e
ChildNodes.
Você pode usar esses métodos de consulta para localizar o argumento para o método Main como uma alternativa
a navegar pela árvore. Adicione o seguinte código à parte inferior do método Main :
WriteLine(argsParameter == argsParameter2);
A primeira instrução usa uma expressão LINQ e o método DescendantNodes para localizar o mesmo parâmetro
do exemplo anterior.
Execute o programa e você poderá ver que a expressão LINQ encontrou o mesmo parâmetro encontrado ao
navegar manualmente pela árvore.
O exemplo usa instruções WriteLine para exibir informações sobre as árvores de sintaxe conforme elas são
percorridas. Você também pode aprender mais executando o programa concluído no depurador. Você pode
examinar mais das propriedades e métodos que fazem parte da árvore de sintaxe criada para o programa Olá,
Mundo.
Caminhadores de sintaxe
Muitas vezes, você deseja localizar todos os nós de um tipo específico em uma árvore de sintaxe, por exemplo,
cada declaração de propriedade em um arquivo. Ao estender a classe
Microsoft.CodeAnalysis.CSharp.CSharpSyntaxWalker e substituir o método
VisitPropertyDeclaration(PropertyDeclarationSyntax), você processa cada declaração de propriedade em uma
árvore de sintaxe sem conhecer a estrutura dele com antecedência. CSharpSyntaxWalker é um tipo específico de
CSharpSyntaxVisitor, que visita recursivamente um nó e cada um dos filhos desse nó.
Este exemplo implementa um CSharpSyntaxWalker que examina uma árvore de sintaxe. Ele coleta diretivas
using que ele constata que não estão importando um namespace System .
Crie um novo projeto de Ferramenta de Análise de Código Autônoma do C#; nomeie-o "SyntaxWalker".
Você pode ver o código concluído para essa amostra no nosso repositório do GitHub. A amostra no GitHub
contém os dois projetos descritos neste tutorial.
Assim como no exemplo anterior, você pode definir uma constante de cadeia de caracteres para conter o texto do
programa que você pretende analisar:
namespace TopLevel
{
using Microsoft;
using System.ComponentModel;
namespace Child1
{
using Microsoft.Win32;
using System.Runtime.InteropServices;
class Foo { }
}
namespace Child2
{
using System.CodeDom;
using Microsoft.CSharp;
class Bar { }
}
}";
Este texto de origem contém diretivas using espalhadas em quatro locais diferentes: o nível de arquivo, no
namespace de nível superior e nos dois namespaces aninhados. Este exemplo destaca um cenário principal para
usar a classe CSharpSyntaxWalker para consultar código. Seria complicado visitar cada nó na árvore de sintaxe de
raiz para encontrar declarações using. Em vez disso, você pode criar uma classe derivada e substituir o método
chamado apenas quando o nó atual na árvore é uma diretiva using. O visitante não realiza nenhum trabalho em
nenhum outro tipo de nó. Esse método único examina cada uma das instruções using e compila uma coleção de
namespaces que não estão no namespace System . Você compila um CSharpSyntaxWalker que examina todas as
instruções using , mas apenas as instruções using .
Agora que você definiu o texto do programa, você precisa criar um SyntaxTree e obter a raiz dessa árvore:
Em seguida, crie uma nova classe. No Visual Studio, escolha Projeto > Adicionar Novo Item. Na caixa de
diálogo Adicionar Novo Item, digite UsingCollector.cs como o nome do arquivo.
Você implementa a funcionalidade de visitante using na classe UsingCollector . Para começar, crie a classe
UsingCollector derivada de CSharpSyntaxWalker.
class UsingCollector : CSharpSyntaxWalker
Você precisa de armazenamento para conter os nós de namespace que você está coletando. Declare uma
propriedade pública somente leitura na classe UsingCollector ; use essa variável para armazenar os nós
UsingDirectiveSyntax que você encontrar:
A classe base CSharpSyntaxWalker implementa a lógica para visitar cada nó na árvore de sintaxe. A classe
derivada substitui os métodos chamados para os nós específicos nos quais você está interessado. Nesse caso, você
está interessado em qualquer diretiva using . Isso significa que você deve substituir o método
VisitUsingDirective(UsingDirectiveSyntax). Um argumento para esse método é um objeto
Microsoft.CodeAnalysis.CSharp.Syntax.UsingDirectiveSyntax. Essa é uma importante vantagem de se usar os
visitantes: eles chamam os métodos substituídos com argumentos que já foram convertidos para o tipo de nó
específico. A classe Microsoft.CodeAnalysis.CSharp.Syntax.UsingDirectiveSyntax tem uma propriedade Name que
armazena o nome do namespace que está sendo importado. É um
Microsoft.CodeAnalysis.CSharp.Syntax.NameSyntax. Adicione o código a seguir na substituição
VisitUsingDirective(UsingDirectiveSyntax):
Assim como no exemplo anterior, você adicionou uma variedade de instruções WriteLine para ajudar na
compreensão do método. Você pode ver quando ele é chamado e quais argumentos são passados para ele a cada
vez.
Por fim, você precisa adicionar duas linhas de código para criar o UsingCollector e fazer com que ele acesse o nó
raiz, coletando todas as instruções using . Em seguida, adicione um loop foreach para exibir todas as instruções
using encontradas pelo seu coletor:
Parabéns! Você usou a API de sintaxe para localizar tipos específicos de instruções C# e declarações em código-
fonte C#.
Introdução à análise semântica
23/10/2019 • 15 minutes to read • Edit Online
Este tutorial presume que você está familiarizado com a API de sintaxe. O artigo Introdução à a análise de sintaxe
fornece uma introdução suficiente.
Neste tutorial, você explora as APIs de Símbolo e de Associação. Essas APIs fornecem informações sobre o
significado semântico de um programa. Elas permitem fazer e responder perguntas sobre os tipos representado
por qualquer símbolo em seu programa.
Você deverá instalar o SDK do .NET Compiler Platform:
Consultar símbolos
Neste tutorial, você analisa novamente o programa "Olá, Mundo". Dessa vez, você consulta os símbolos no
programa para compreender quais tipos esses símbolos representam. Você consulta os tipos em um namespace e
aprende a localizar os métodos disponíveis em um tipo.
Você pode ver o código concluído para essa amostra no nosso repositório do GitHub.
NOTE
Os tipos de árvore de sintaxe usam a herança para descrever os elementos de sintaxe diferentes que são válidos em locais
diferentes no programa. Usar essas APIs geralmente significa converter propriedades ou membros da coleção em tipos
derivados específicos. Nos exemplos a seguir, a atribuição e as conversões são instruções separadas, usando variáveis
explicitamente tipadas. Você pode ler o código para ver os tipos de retorno da API e o tipo de tempo de execução dos
objetos retornados. Na prática, é mais comum usar variáveis implicitamente tipadas e depender de nomes de API para
descrever o tipo de objeto que está sendo examinado.
namespace HelloWorld
{
class Program
{
static void Main(string[] args)
{
Console.WriteLine(""Hello, World!"");
}
}
}";
Em seguida, adicione o código a seguir para criar a árvore de sintaxe para o texto do código na constante
programText . Adicione a seguinte linha ao seu método Main :
Em seguida, compile uma CSharpCompilation da árvore que você já criou. A amostra "Olá, Mundo" depende dos
tipos String e Console. Você precisa fazer referência ao assembly que declara esses dois tipos em sua compilação.
Adicione a seguinte linha ao seu método Main para criar uma compilação de sua árvore de sintaxe, incluindo a
referência ao assembly apropriado:
Associar um nome
A Compilation cria o SemanticModel da SyntaxTree. Depois de criar o modelo, você pode consultar para localizar a
primeira diretiva using e recuperar as informações de símbolo para o namespace System . Adicione estas duas
linhas a seu método Main para criar o modelo semântico e recuperar o símbolo para a primeira instrução using:
O código anterior mostra como associar o nome na primeira diretiva using para recuperar um
Microsoft.CodeAnalysis.SymbolInfo para o namespace System . O código anterior também ilustra o uso da
sintaxe de modelo para localizar a estrutura do código; você usa o modelo semântico para entender seu
significado. A sintaxe de modelo localiza a cadeia de caracteres System na instrução using. O modelo
semântico tem todas as informações sobre os tipos definidos no namespace System .
Do objeto SymbolInfo, você pode obter o Microsoft.CodeAnalysis.ISymbol usando a propriedade
SymbolInfo.Symbol. Essa propriedade retorna o símbolo a que essa expressão se refere. Para expressões que não
se referem a nada (como literais numéricos), essa propriedade é null . Quando o SymbolInfo.Symbol não for null,
o ISymbol.Kind denotará o tipo do símbolo. Nesse exemplo, a propriedade ISymbol.Kind é um
SymbolKind.Namespace. Adicione o código a seguir ao método Main . Ele recupera o símbolo para o namespace
System e, em seguida, exibe todos os namespaces filho declarados no namespace System :
System.Collections
System.Configuration
System.Deployment
System.Diagnostics
System.Globalization
System.IO
System.Numerics
System.Reflection
System.Resources
System.Runtime
System.Security
System.StubHelpers
System.Text
System.Threading
Press any key to continue . . .
NOTE
A saída não inclui todos os namespaces que são namespaces filhos do namespace System . El exibe cada namespace
presente nessa compilação, que só referencia o assembly em que System.String é declarada. Quaisquer outros
namespaces declarados em outros assemblies não são conhecidos desta compilação
Para concluir este tutorial, criaremos uma consulta LINQ que criará uma sequência de todos os métodos públicos
declarados no tipo string que retorna um string . Essa consulta torna-se complexa, então a compilaremos linha
a linha e então a reconstruiremos como uma única consulta. A ordem desta consulta é a sequência de todos os
membros declarados no tipo string :
Essa sequência de origem contém todos os membros, incluindo propriedades e campos, portanto, filtre-a usando
o método ImmutableArray<T>.OfType para localizar elementos que são objetos
Microsoft.CodeAnalysis.IMethodSymbol:
Em seguida, adicione outro filtro para retornar somente os métodos que são públicos e retornam um string :
Selecione apenas a propriedade de nome e somente os nomes distintos, removendo quaisquer sobrecargas:
Você pode também compilar a consulta completa usando a sintaxe de consulta LINQ e, em seguida, exibir todos
os nomes de método no console:
Você usou a API de semântica para localizar e exibir informações sobre os símbolos que fazem parte deste
programa.
Introdução à transformação de sintaxe
31/10/2019 • 23 minutes to read • Edit Online
Este tutorial baseia-se nos conceitos e técnicas explorados nos guias de início rápido Introdução à análise de
sintaxe e Introdução à análise semântica. Se ainda não o fez, você deve concluir as etapas rápidas antes de começar
esta.
Neste início rápido, você explora técnicas para criar e transformar árvores de sintaxe. Em combinação com as
técnicas que você aprendeu em guias de início rápido anteriores, você cria sua primeira refatoração de linha de
comando!
Você criará nós de sintaxe de nome para construir a árvore que representa a instrução
using System.Collections.Generic; . NameSyntax é a classe base para quatro tipos de nomes que aparecem no C#.
Você compõe esses quatro tipos de nomes para criar qualquer nome que possa aparecer na linguagem C#:
Microsoft.CodeAnalysis.CSharp.Syntax.NameSyntax, que representa nomes simples de identificadores únicos
como System e Microsoft .
Microsoft.CodeAnalysis.CSharp.Syntax.GenericNameSyntax, que representa um tipo genérico ou nome de
método, como List<int> .
Microsoft.CodeAnalysis.CSharp.Syntax.QualifiedNameSyntax,que representa um nome qualificado do
formulário <left-name>.<right-identifier-or-generic-name> , como System.IO .
Microsoft.CodeAnalysis.CSharp.Syntax.AliasQualifiedNameSyntax, que representa um nome usando um alias
externo de assembly como LibraryV2::Foo .
Você usa o método IdentifierName(String) para criar um nó NameSyntax. Adicione o seguinte código no seu
método Main no Program.cs :
O código anterior cria um objeto IdentifierNameSyntax e o atribui à variável name . Muitas das APIs Roslyn
retornam classes básicas para facilitar o trabalho com tipos relacionados. A variável name , um NameSyntax, pode
ser reutilizada conforme você constrói o QualifiedNameSyntax. Não use inferência de tipo ao criar a amostra. Você
automatizará essa etapa neste projeto.
Você criou o nome. Agora, é hora de criar mais nós na árvore criando um QualifiedNameSyntax. A nova árvore
usa name como a esquerda do nome e um novo IdentifierNameSyntax para o namespace Collections como o
lado direito do QualifiedNameSyntax. Adicione o seguinte código ao program.cs :
Execute o código novamente e confira os resultados. Você está construindo uma árvore de nós que representa o
código. Você continuará este padrão para construir o QualifiedNameSyntax para o namespace
System.Collections.Generic . Adicione o seguinte código ao Program.cs :
Execute o programa novamente para ver que você construiu a árvore para o código a ser adicionado.
Criar uma árvore modificada
Você criou uma pequena árvore de sintaxe que contém uma instrução. As APIs para criar novos nós são a escolha
certa para criar instruções únicas ou outros pequenos blocos de código. No entanto, para construir blocos maiores
de código, você deve usar métodos que substituem nós ou inserem nós em uma árvore existente. Lembre-se de
que as árvores de sintaxe são imutáveis. A API de Sintaxe não fornece nenhum mecanismo para modificar uma
árvore de sintaxe existente após a construção. Em vez disso, fornece métodos que produzem novas árvores com
base nas alterações existentes. Os métodos With* são definidos em classes concretas que derivam de SyntaxNode
ou em métodos de extensão declarados na classe SyntaxNodeExtensions. Esses métodos criam um novo nó
aplicando alterações nas propriedades filho de um nó existente. Além disso, o método de extensão ReplaceNode
pode ser usado para substituir um nó descendente em uma subárvore. Esse método também atualiza o pai para
apontar para o filho recém-criado e repete esse processo até a árvore inteira — um processo conhecido como re-
spinning da árvore.
O próximo passo é criar uma árvore que represente um programa inteiro (pequeno) e depois modificá-la. Adicione
o seguinte código ao início da classe Program :
namespace HelloWorld
{
class Program
{
static void Main(string[] args)
{
Console.WriteLine(""Hello, World!"");
}
}
}";
NOTE
O código de exemplo usa o namespace System.Collections e não o namespace System.Collections.Generic .
Em seguida, adicione o seguinte código à parte inferior do método Main para analisar o texto e criar uma árvore:
Este exemplo usa o método WithName(NameSyntax) para substituir o nome em um nó UsingDirectiveSyntax pelo
que foi construído no código anterior.
Crie um novo nó UsingDirectiveSyntax usando o método WithName(NameSyntax) para atualizar o nome
System.Collections com o nome criado no código anterior. Adicione o seguinte código à parte inferior do método
Main :
Execute o programa e observe atentamente a saída. O newusing não foi colocado na árvore raiz. A árvore original
não foi alterada.
Adicione o seguinte código usando o método de extensão ReplaceNode para criar uma nova árvore. A nova árvore
é o resultado da substituição da importação existente pelo nó newUsing atualizado. Você atribui essa nova árvore à
variável root existente:
Os métodos With* e ReplaceNode fornecem meios convenientes para transformar ramos individuais de uma
árvore de sintaxe. A classe Microsoft.CodeAnalysis.CSharp.CSharpSyntaxRewriter realiza várias transformações
em uma árvore de sintaxe. A classe Microsoft.CodeAnalysis.CSharp.CSharpSyntaxRewriter é uma subclasse de
Microsoft.CodeAnalysis.CSharp.CSharpSyntaxVisitor<TResult>. O CSharpSyntaxRewriter aplica uma
transformação a um tipo específico de SyntaxNode. Você pode aplicar transformações a vários tipos de objetos
SyntaxNode sempre que eles aparecerem em uma árvore de sintaxe. O segundo projeto neste guia de início rápido
cria uma refatoração de linha de comando que remove tipos explícitos em declarações de variáveis locais em
qualquer lugar em que a inferência de tipo possa ser usada.
Crie um novo projeto de Ferramenta de Análise de Código Autônoma do C#. No Visual Studio, clique com o
botão direito do mouse no nó da solução SyntaxTransformationQuickStart . Escolha Adicionar > Novo Projeto
para exibir o diálogo Novo Projeto. Em Visual C# > Extensibilidade, escolha Ferramenta de Análise de
Código Autônoma. Nomeie seu projeto como TransformationCS e clique em OK.
A primeira etapa é criar uma classe que deriva de CSharpSyntaxRewriter para executar as transformações.
Adicione um novo arquivo de classe ao projeto. No Visual Studio, escolha Projeto > Adicionar Classe... . Na
caixa de diálogo Adicionar Novo Item, digite TypeInferenceRewriter.cs como o nome do arquivo.
Adicione o seguinte usando diretivas ao arquivo TypeInferenceRewriter.cs :
using Microsoft.CodeAnalysis;
using Microsoft.CodeAnalysis.CSharp;
using Microsoft.CodeAnalysis.CSharp.Syntax;
using static Microsoft.CodeAnalysis.CSharp.SyntaxFactory;
Adicione o seguinte código para declarar um campo somente leitura privado para conter um SemanticModel e
inicializá-lo no construtor. Você precisará deste campo posteriormente para determinar onde a inferência de tipos
pode ser usada:
NOTE
Muitas das APIs Roslyn declaram os tipos de retorno que são classes base dos tipos de tempo de execução reais retornados.
Em muitos cenários, um tipo de nó pode ser substituído inteiramente por outro tipo de nó, ou até mesmo removido. Neste
exemplo, o método VisitLocalDeclarationStatement(LocalDeclarationStatementSyntax) retorna um SyntaxNode, em vez do
tipo derivado de LocalDeclarationStatementSyntax. Este regravador retorna um novo nó LocalDeclarationStatementSyntax
baseado no existente.
Este guia de início rápido lida com declarações de variáveis locais. Você poderia estendê-lo para outras declarações,
como loops foreach , loops for , expressões LINQ e expressões lambda. Além disso, este regravador só irá
transformar as declarações da forma mais simples:
Se você quiser explorar por conta própria, considere estender a amostra finalizada para esses tipos de declarações
de variáveis:
Adicione o seguinte código ao corpo do método VisitLocalDeclarationStatement para ignorar a reescrita dessas
formas de declaração:
if (node.Declaration.Variables.Count > 1)
{
return node;
}
if (node.Declaration.Variables[0].Initializer == null)
{
return node;
}
O método indica que nenhuma reescrita ocorre retornando o parâmetro node não modificado. Se nenhuma
dessas expressões if for verdadeira, o nó representa uma possível declaração com inicialização. Adicione estas
instruções para extrair o nome do tipo especificado na declaração e vinculá-lo usando o campo SemanticModel
para obter um símbolo de tipo:
Por fim, inclua a seguinte instrução if para substituir o nome do tipo existente pela palavra-chave var , se o tipo
de expressão do inicializador corresponder ao tipo especificado:
if (variableType == initializerInfo.Type)
{
TypeSyntax varTypeName = IdentifierName("var")
.WithLeadingTrivia(variableTypeName.GetLeadingTrivia())
.WithTrailingTrivia(variableTypeName.GetTrailingTrivia());
A condicional é necessária porque a declaração pode converter a expressão inicializadora em uma classe ou
interface base. Se este for o caso, os tipos à esquerda e à direita da atribuição não coincidem. Remover o tipo
explícito nesses casos alteraria a semântica de um programa. var é especificado como um identificador em vez de
uma palavra-chave porque var é uma palavra-chave contextual. As trivialidades inicial e final (espaço em branco)
são transferidas do nome do tipo antigo para a palavra-chave var para manter espaço em branco vertical e recuo.
É mais simples usar ReplaceNode em vez de With* para transformar o LocalDeclarationStatementSyntax, pois o
nome do tipo é na verdade o neto da declaração.
Você concluiu o TypeInferenceRewriter . Agora, retorne ao arquivo Program.cs para finalizar o exemplo. Crie um
teste Compilation e obtenha o SemanticModel dele. Use esse SemanticModel para testar seu
TypeInferenceRewriter . Você realizará esta etapa por último. Enquanto isso, declare uma variável de espaço
reservado representando sua compilação de teste:
Compilation test = CreateTestCompilation();
Após uma pausa, um erro será exibido informando que não existe método CreateTestCompilation . Pressione Ctrl
+ Ponto para abrir a lâmpada e pressione Enter para invocar o comando Gerar Stub de Método. Este comando
irá gerar um stub de método para o método CreateTestCompilation na classe Program . Você voltará a preencher
este método mais tarde:
Grave o seguinte código para iterar sobre cada SyntaxTree no teste Compilation. Para cada um, inicialize um novo
TypeInferenceRewriter com o SemanticModel para essa árvore:
if (newSource != sourceTree.GetRoot())
{
File.WriteAllText(sourceTree.FilePath, newSource.ToFullString());
}
}
Dentro da instrução foreach criada, adicione o seguinte código para executar a transformação em cada árvore de
origem. Este código registra condicionalmente a nova árvore transformada se alguma edição tiver sido feita. Seu
regravador só deve modificar uma árvore se encontrar uma ou mais declarações de variáveis locais que poderiam
ser simplificadas usando a inferência de tipos:
if (newSource != sourceTree.GetRoot())
{
File.WriteAllText(sourceTree.FilePath, newSource.ToFullString());
}
Você deve ver rabiscos abaixo do código File.WriteAllText . Selecione a lâmpada e adicione a instrução
using System.IO; necessária.
Você está quase lá! Resta somente uma etapa: criar um teste Compilation. Como você não usou a inferência de
tipos durante este guia de início rápido, este seria um caso de teste perfeito. Infelizmente, criar uma compilação a
partir de um arquivo de projeto C# está além do escopo desta explicação passo a passo. Mas, felizmente, se você
está seguindo as instruções cuidadosamente, há esperança. Substitua o conteúdo do método de
CreateTestCompilation com o código a seguir. Ele cria uma compilação de teste que combina coincidentemente
com o projeto descrito neste guia de início rápido:
MetadataReference mscorlib =
MetadataReference.CreateFromFile(typeof(object).Assembly.Location);
MetadataReference codeAnalysis =
MetadataReference.CreateFromFile(typeof(SyntaxTree).Assembly.Location);
MetadataReference csharpCodeAnalysis =
MetadataReference.CreateFromFile(typeof(CSharpSyntaxTree).Assembly.Location);
return CSharpCompilation.Create("TransformationCS",
sourceTrees,
references,
new CSharpCompilationOptions(OutputKind.ConsoleApplication));
Cruze os dedos e execute o projeto. No Visual Studio, escolha Depurar > Iniciar Depuração. O Visual Studio
exibirá um aviso dizendo que os arquivos em seu projeto foram alterados. Clique em "Sim para Todos" para
recarregar os arquivos modificados. Examine-os para observar sua grandiosidade. Observe como o código parece
mais limpo sem todos os especificadores de tipo explícitos e redundantes.
Parabéns! Você usou as APIs do compilador para criar sua própria refatoração que pesquisa todos os arquivos
em um projeto C# para determinados padrões sintáticos, analisa a semântica do código-fonte que corresponde a
esses padrões e os transforma. Agora você é oficialmente um autor de refatoração!
Tutorial: crie seu primeiro analisador e correção de
código
23/10/2019 • 46 minutes to read • Edit Online
O SDK da .NET Compiler Platform fornece as ferramentas necessárias para criar avisos personalizados destinados
ao C# ou código do Visual Basic. Seu analisador contém código que reconhece as violações da sua regra. Sua
correção de código contém o código que corrige a violação. As regras que você implementar podem ser
qualquer coisa, incluindo estrutura do código, estilo de codificação, convenções de nomenclatura e muito mais. O
.NET Compiler Platform fornece a estrutura para executar análise conforme os desenvolvedores escrevem o
código, bem como todos os recursos de interface do usuário do Visual Studio para corrigir o código: mostrar
rabiscos no editor, popular a Lista de Erros do Visual Studio, criar as sugestões da "lâmpada" e mostrar a
visualização avançada das correções sugeridas.
Neste tutorial, você explorará a criação de um analisador e uma correção de código que o acompanha, usando
as APIs do Roslyn. Um analisador é uma maneira de executar a análise de código-fonte e relatar um problema para
o usuário. Opcionalmente, um analisador também pode fornecer uma correção de código que representa uma
modificação no código-fonte do usuário. Este tutorial cria um analisador que localiza as declarações de variável
local que poderiam ser declaradas usando o modificador const , mas não o são. A correção de código anexa
modifica essas declarações para adicionar o modificador const .
Pré-requisitos
Visual Studio 2017
Visual Studio 2019
Será necessário instalar o SDK do .NET Compiler Platform por meio do Instalador do Visual Studio:
int x = 0;
Console.WriteLine(x);
No código acima, um valor constante é atribuído a x , que nunca é modificado. Ele pode ser declarado usando o
modificador const :
const int x = 0;
Console.WriteLine(x);
A análise para determinar se uma variável pode ser tornada constante está envolvida, exigindo análise sintática,
análise constante da expressão de inicializador e também análise de fluxo de dados, para garantir que nunca
ocorram gravações na variável. O .NET Compiler Platform fornece APIs que facilitam essa análise. A primeira
etapa é criar um novo em projeto de Analisador com correção de código do C#.
No Visual Studio, escolha Arquivo > Novo > Projeto... para exibir a caixa de diálogo Novo Projeto.
Em Visual C# > Extensibilidade, escolha Analisador com correção de código (.NET Standard) .
Nomeie seu projeto como "MakeConst" e clique em OK.
O analisador com modelo de correção de código cria três projetos: um contém o analisador e a correção de código,
o segundo é um projeto de teste de unidade e o terceiro é o projeto VSIX. O projeto de inicialização padrão é o
projeto VSIX. Pressione F5 para iniciar o projeto VSIX. Isso inicia uma segunda instância do Visual Studio que
tenha carregado o seu novo analisador.
TIP
Quando você executa seu analisador, você pode iniciar uma segunda cópia do Visual Studio. Essa segunda cópia usa um hive
do Registro diferente para armazenar configurações. Isso lhe permite diferenciar as configurações visuais em duas cópias do
Visual Studio. Você pode escolher um tema diferente para a execução experimental do Visual Studio. Além disso, não use
perfil móvel de suas configurações nem faça logon na conta do Visual Studio usando a execução experimental do Visual
Studio. Isso mantém as diferenças entre as configurações.
Na segunda instância do Visual Studio que você acabou de iniciar, crie um novo projeto de Aplicativo de Console
em C# (uma das opções entre o projeto do .NET Core e o do .NET Framework funcionará – os analisadores
trabalham no nível da origem.) Passe o mouse sobre o token com um sublinhado ondulado e o texto de aviso
fornecido por um analisador será exibido.
O modelo cria um analisador que relata um aviso em cada declaração de tipo em que o nome do tipo contém
letras minúsculas, conforme mostrado na figura a seguir:
O modelo também fornece uma correção de código que altera qualquer nome de tipo que contenha caracteres de
letras minúsculas, deixando-o com todas as letras maiúsculas. Você pode clicar na lâmpada exibida com o aviso
para ver as alterações sugeridas. Aceitar as alterações sugeridas atualiza o nome do tipo e todas as referências para
esse tipo na solução. Agora que você já viu o analisador inicial em ação, feche a segunda instância do Visual Studio
e retorne ao projeto do analisador.
Você não precisa iniciar uma segunda cópia do Visual Studio e criar um novo código para testar cada alteração em
seu analisador. O modelo também cria um projeto de teste de unidade para você. Esse projeto contém dois testes.
TestMethod1 mostra o formato típico de um teste que analisa o código sem disparar um diagnóstico. TestMethod2
mostra o formato de um teste que dispara um diagnóstico e, em seguida, aplica uma correção de código sugerida.
Conforme você cria o analisador e a correção de código, você escreve testes para estruturas de código diferentes
para verificar seu trabalho. Testes de unidade para os analisadores são muito mais rápidos do que testá-los de
forma interativa com o Visual Studio.
TIP
Testes de unidade de analisador são uma excelente ferramenta quando você sabe quais constructos de código devem e não
devem disparar seu analisador. Carregar o analisador em outra cópia do Visual Studio é uma excelente ferramenta para
explorar e encontrar constructos nos quais você talvez não tenha pensado ainda.
Além disso, lembre-se de alterar a lista suspensa Modificador de Acesso para public . Isso facilita o uso dessas
constantes em testes de unidade. Quando você terminar, o editor de recursos deverá aparecer como mostrado na
figura a seguir:
As alterações restantes estão no arquivo do analisador. Abra MakeConstAnalyzer.cs no Visual Studio. Altere a
ação registrada de uma que age em símbolos para uma que age sobre a sintaxe. No método
MakeConstAnalyzerAnalyzer.Initialize , localize a linha que registra a ação em símbolos:
context.RegisterSymbolAction(AnalyzeSymbol, SymbolKind.NamedType);
context.ConfigureGeneratedCodeAnalysis(GeneratedCodeAnalysisFlags.Analyze | GeneratedCodeAnalysisFlags.None);
context.EnableConcurrentExecution();
context.RegisterSyntaxNodeAction(AnalyzeNode, SyntaxKind.LocalDeclarationStatement);
Após essa alteração, você poderá excluir o método AnalyzeSymbol . Este analisador examina
SyntaxKind.LocalDeclarationStatement, e não instruções SymbolKind.NamedType. Observe que AnalyzeNode tem
rabiscos vermelhos sob ele. O código apenas que você acaba de adicionar referencia um método AnalyzeNode que
não foi declarado. Declare esse método usando o seguinte código:
Essa conversão sempre terá êxito porque seu analisador fez o registro para alterações unicamente a declarações
locais. Nenhum outro tipo de nó dispara uma chamada para seu método AnalyzeNode . Em seguida, verifique a
declaração para quaisquer modificadores const . Se você encontrá-los, retorne imediatamente. O código a seguir
procura por quaisquer modificadores const na declaração local:
Por fim, você precisa verificar que a variável pode ser const . Isso significa assegurar que ela nunca seja atribuída
após ser inicializada.
Você executará alguma análise semântica usando o SyntaxNodeAnalysisContext. Você usa o argumento context
para determinar se a declaração de variável local pode ser tornada const . Um
Microsoft.CodeAnalysis.SemanticModel representa de todas as informações semânticas em um único arquivo de
origem. Você pode aprender mais no artigo que aborda modelos semânticos. Você usará o
Microsoft.CodeAnalysis.SemanticModel para realizar a análise de fluxo de dados na instrução de declaração local.
Em seguida, você usa os resultados dessa análise de fluxo de dados para garantir que a variável local não seja
escrita com um novo valor em nenhum outro lugar. Chame o método de extensão GetDeclaredSymbol para
recuperar o ILocalSymbol para a variável e verifique se ele não está contido na coleção
DataFlowAnalysis.WrittenOutside da análise de fluxo de dados. Adicione o código a seguir ao final do método
AnalyzeNode :
// Retrieve the local symbol for each variable in the local declaration
// and ensure that it is not written outside of the data flow analysis region.
var variable = localDeclaration.Declaration.Variables.Single();
var variableSymbol = context.SemanticModel.GetDeclaredSymbol(variable);
if (dataFlowAnalysis.WrittenOutside.Contains(variableSymbol))
{
return;
}
O código recém-adicionado garante que a variável não seja modificada e pode, portanto, ser tornada const .É
hora de gerar o diagnóstico. Adicione o código a seguir como a última linha em AnalyzeNode :
context.ReportDiagnostic(Diagnostic.Create(Rule, context.Node.GetLocation()));
Você pode verificar seu andamento pressionando F5 para executar o analisador. Você pode carregar o aplicativo de
console que você criou anteriormente e, em seguida, adicionar o seguinte código de teste:
int x = 0;
Console.WriteLine(x);
A lâmpada deve aparecer e o analisador deve relatar um diagnóstico. No entanto, a lâmpada ainda usa a correção
de código gerada por modelo, e diz a você que ela pode ser colocada em letras maiúsculas. A próxima seção explica
como escrever a correção de código.
const int x = 0;
Console.WriteLine(x);
O usuário escolhe-a da lâmpada da interface do usuário no editor e do Visual Studio altera o código.
Abra o arquivo MakeConstCodeFixProvider.cs adicionado pelo modelo. Essa correção de código já está
conectada à ID de Diagnóstico produzida pelo analisador de diagnóstico, mas ela ainda não implementa a
transformação de código correta. Primeiro você deve remover parte do código de modelo. Altere a cadeia de
caracteres do título para "Tornar constante":
var declaration =
root.FindToken(diagnosticSpan.Start).Parent.AncestorsAndSelf().OfType<LocalDeclarationStatementSyntax>
().First();
Em seguida, altere a última linha para registrar uma correção de código. A correção criará um novo documento
resultante da adição do modificador const para uma declaração existente:
Você observará rabiscos vermelhos no código que você acabou de adicionar no símbolo MakeConstAsync . Adicione
uma declaração para MakeConstAsync semelhante ao seguinte código:
private async Task<Document> MakeConstAsync(Document document,
LocalDeclarationStatementSyntax localDeclaration,
CancellationToken cancellationToken)
{
}
Seu novo método MakeConstAsync transformará o Document que representa o arquivo de origem do usuário em
um novo Document que agora contém uma declaração const .
Você cria um novo token de palavra-chave const a ser inserido no início da instrução de declaração. Tenha
cuidado para remover qualquer desafio à esquerda do primeiro token de instrução de declaração e anexe-o ao
token const . Adicione o seguinte código ao método MakeConstAsync :
// Insert the const token into the modifiers list, creating a new modifiers list.
var newModifiers = trimmedLocal.Modifiers.Insert(0, constToken);
// Produce the new local declaration.
var newLocal = trimmedLocal
.WithModifiers(newModifiers)
.WithDeclaration(localDeclaration.Declaration);
Em seguida, formate a nova declaração de acordo com regras de formatação de C#. Formatar de suas alterações
para corresponderem ao código existente cria uma experiência melhor. Adicione a instrução a seguir
imediatamente após o código existente:
Um novo namespace é necessário para esse código. Adicione a instrução using a seguir à parte superior do
arquivo:
using Microsoft.CodeAnalysis.Formatting;
A correção de código está pronta para ser experimentada. Pressione F5 para executar o projeto do analisador em
uma segunda instância do Visual Studio. Na segunda instância do Visual Studio, crie um novo projeto de Aplicativo
de Console em C# e adicione algumas declarações de variável local inicializadas com valores constantes para o
método Main. Você verá que elas são relatadas como avisos, conforme mostrado a seguir.
Você fez muito progresso. Há rabiscos sob as declarações que podem ser tornados const . Mas ainda há trabalho a
fazer. Isso funciona bem se você adicionar const às declarações começando com i , depois j e, por fim, k . Mas
se você adicionar o modificador const em uma ordem diferente, começando com k , seu analisador criará erros:
k não pode ser declarado como const , a menos que i e j já sejam ambos const . Você tem que fazer mais
análise para assegurar que lida com as diferentes maneiras em que variáveis podem ser declaradas e inicializadas.
O código para quase todos os testes para o seu analisador segue um destes dois padrões. Para a primeira etapa,
você pode refazer esses testes como testes controlados por dados. Em seguida, será fácil criar novos testes
adicionando novas constantes de cadeia de caracteres para representar diferentes entradas de teste.
Para obter eficiência, a primeira etapa é refatorar os dois testes em testes controlados por dados. Em seguida, você
só precisa definir algumas constantes de cadeia de caracteres para cada novo teste. Durante a refatoração,
renomeie os dois métodos com nomes melhores. Substitua TestMethod1 com este teste, que garante que nenhum
diagnóstico seja gerado:
[DataTestMethod]
[DataRow("")]
public void WhenTestCodeIsValidNoDiagnosticIsTriggered(string testCode)
{
VerifyCSharpDiagnostic(testCode);
}
Você pode criar uma nova linha de dados para este teste, definindo qualquer fragmento de código que não faça
com que o diagnóstico dispare um aviso. Essa sobrecarga de VerifyCSharpDiagnostic passa quando não há
nenhum diagnóstico disparado para o fragmento de código-fonte.
Em seguida, substitua TestMethod2 com esse teste que garante que um diagnóstico é gerado e uma correção de
código aplicada ao fragmento de código-fonte:
[DataTestMethod]
[DataRow(LocalIntCouldBeConstant, LocalIntCouldBeConstantFixed, 10, 13)]
public void WhenDiagnosticIsRaisedFixUpdatesCode(
string test,
string fixTest,
int line,
int column)
{
var expected = new DiagnosticResult
{
Id = MakeConstAnalyzer.DiagnosticId,
Message = new LocalizableResourceString(nameof(MakeConst.Resources.AnalyzerMessageFormat),
MakeConst.Resources.ResourceManager, typeof(MakeConst.Resources)).ToString(),
Severity = DiagnosticSeverity.Warning,
Locations =
new[] {
new DiagnosticResultLocation("Test0.cs", line, column)
}
};
VerifyCSharpDiagnostic(test, expected);
VerifyCSharpFix(test, fixTest);
}
O código anterior também fez algumas alterações no código que cria o resultado de diagnóstico esperado. Ele usa
as constantes públicas registradas no analisador MakeConst . Além disso, ele usa duas constantes de cadeia de
caracteres para a fonte de entrada e fonte fixa. Adicione as seguintes constantes de cadeia de caracteres à classe
UnitTest :
namespace MakeConstTest
{
class Program
{
static void Main(string[] args)
{
int i = 0;
Console.WriteLine(i);
}
}
}";
namespace MakeConstTest
{
class Program
{
static void Main(string[] args)
{
const int i = 0;
Console.WriteLine(i);
}
}
}";
Execute esses dois testes para certificar-se de sua aprovação. No Visual Studio, abra o Gerenciador de Testes
selecionando Teste > Windows > Gerenciador de Testes. Em seguida, pressione o link Executar Tudo.
namespace MakeConstTest
{
class Program
{
static void Main(string[] args)
{
int i = 0;
Console.WriteLine(i++);
}
}
}";
Em seguida, adicione uma linha de dados para este teste, conforme mostrado no snippet de código a seguir:
[DataTestMethod]
[DataRow(""),
DataRow(VariableAssigned)]
public void WhenTestCodeIsValidNoDiagnosticIsTriggered(string testCode)
Esse teste é aprovado também. Em seguida, adicione as constantes para as condições que você ainda não
manipulou ainda:
Declarações que já são const , porque elas já são const:
namespace MakeConstTest
{
class Program
{
static void Main(string[] args)
{
const int i = 0;
Console.WriteLine(i);
}
}
}";
Declarações que não têm nenhum inicializador, porque não há nenhum valor a ser usado:
private const string NoInitializer = @"
using System;
namespace MakeConstTest
{
class Program
{
static void Main(string[] args)
{
int i;
i = 0;
Console.WriteLine(i);
}
}
}";
Declarações em que o inicializador não é uma constante, porque elas não podem ser constantes de tempo
de compilação:
namespace MakeConstTest
{
class Program
{
static void Main(string[] args)
{
int i = DateTime.Now.DayOfYear;
Console.WriteLine(i);
}
}
}";
Isso pode ser ainda mais complicado, porque o C# permite várias declarações como uma instrução. Considere a
seguinte constante de cadeia de caracteres de caso de teste:
namespace MakeConstTest
{
class Program
{
static void Main(string[] args)
{
int i = 0, j = DateTime.Now.DayOfYear;
Console.WriteLine(i, j);
}
}
}";
A variável i pode ser tornada constante, mas o mesmo não se aplica à variável j . Portanto, essa instrução não
pode ser tornada uma declaração const. Adicione as declarações DataRow para todos esses testes:
[DataTestMethod]
[DataRow(""),
DataRow(VariableAssigned),
DataRow(AlreadyConst),
DataRow(NoInitializer),
DataRow(InitializerNotConstant),
DataRow(MultipleInitializers)]
public void WhenTestCodeIsValidNoDiagnosticIsTriggered(string testCode)
Execute os testes novamente e você verá esses novos casos de teste falharem.
// Retrieve the local symbol for each variable in the local declaration
// and ensure that it is not written outside of the data flow analysis region.
var variable = localDeclaration.Declaration.Variables.Single();
var variableSymbol = context.SemanticModel.GetDeclaredSymbol(variable);
if (dataFlowAnalysis.WrittenOutside.Contains(variableSymbol))
{
return;
}
O primeiro loop de foreach examina cada declaração de variável usando a análise sintática. A primeira verificação
garante que a variável tenha um inicializador. A segunda verificação garante que o inicializador seja uma constante.
O segundo loop tem a análise semântica original. As verificações semânticas estão em um loop separado porque
ele tem um impacto maior no desempenho. Execute os testes novamente e você deverá ver todos eles serem
aprovados.
namespace MakeConstTest
{
class Program
{
static void Main(string[] args)
{
int x = ""abc"";
}
}
}";
Além disso, os tipos de referência não são tratados corretamente. O único valor de constante permitido para um
tipo de referência é null , exceto no caso de String, que permite literais de cadeia de caracteres. Em outras
palavras, const string s = "abc" é legal, mas const object s = "abc" não é. Este snippet de código verifica essa
condição:
namespace MakeConstTest
{
class Program
{
static void Main(string[] args)
{
object s = ""abc"";
}
}
}";
Para ser criterioso, você precisará adicionar outro teste para verificar se pode criar uma declaração de constante
para uma cadeia de caracteres. O snippet de código a seguir define o código que gera o diagnóstico e o código
após a aplicação da correção:
namespace MakeConstTest
{
class Program
{
static void Main(string[] args)
{
string s = ""abc"";
}
}
}";
namespace MakeConstTest
{
class Program
{
static void Main(string[] args)
{
const string s = ""abc"";
}
}
}";
Por fim, se uma variável é declarada com a palavra-chave var , a correção de código faz a coisa errada e gera uma
declaração const var , que não é compatível com a linguagem C#. Para corrigir esse bug, a correção de código
deve substituir a palavra-chave var pelo nome do tipo inferido:
private const string DeclarationUsesVar = @"
using System;
namespace MakeConstTest
{
class Program
{
static void Main(string[] args)
{
var item = 4;
}
}
}";
namespace MakeConstTest
{
class Program
{
static void Main(string[] args)
{
const int item = 4;
}
}
}";
private const string StringDeclarationUsesVar = @"
using System;
namespace MakeConstTest
{
class Program
{
static void Main(string[] args)
{
var item = ""abc"";
}
}
}";
private const string StringDeclarationUsesVarFixedHasType = @"
using System;
namespace MakeConstTest
{
class Program
{
static void Main(string[] args)
{
const string item = ""abc"";
}
}
}";
Essas alterações atualizam as declarações de linha de dados para ambos os testes. O código a seguir mostra esses
testes com todos os atributos de linha de dados:
//No diagnostics expected to show up
[DataTestMethod]
[DataRow(""),
DataRow(VariableAssigned),
DataRow(AlreadyConst),
DataRow(NoInitializer),
DataRow(InitializerNotConstant),
DataRow(MultipleInitializers),
DataRow(DeclarationIsInvalid),
DataRow(ReferenceTypeIsntString)]
public void WhenTestCodeIsValidNoDiagnosticIsTriggered(string testCode)
{
VerifyCSharpDiagnostic(testCode);
}
[DataTestMethod]
[DataRow(LocalIntCouldBeConstant, LocalIntCouldBeConstantFixed, 10, 13),
DataRow(ConstantIsString, ConstantIsStringFixed, 10, 13),
DataRow(DeclarationUsesVar, DeclarationUsesVarFixedHasType, 10, 13),
DataRow(StringDeclarationUsesVar, StringDeclarationUsesVarFixedHasType, 10, 13)]
public void WhenDiagosticIsRaisedFixUpdatesCode(
string test,
string fixTest,
int line,
int column)
{
var expected = new DiagnosticResult
{
Id = MakeConstAnalyzer.DiagnosticId,
Message = new LocalizableResourceString(nameof(MakeConst.Resources.AnalyzerMessageFormat),
MakeConst.Resources.ResourceManager, typeof(MakeConst.Resources)).ToString(),
Severity = DiagnosticSeverity.Warning,
Locations =
new[] {
new DiagnosticResultLocation("Test0.cs", line, column)
}
};
VerifyCSharpDiagnostic(test, expected);
VerifyCSharpFix(test, fixTest);
}
Felizmente, todos os erros acima podem ser resolvidos usando as mesmas técnicas que você acabou de aprender.
Para corrigir o primeiro bug, primeiro abra DiagnosticAnalyzer.cs e localize o loop foreach em que cada um dos
inicializadores de declaração local é verificado para garantir que valores constantes sejam atribuídos a eles.
Imediatamente antes do primeiro loop foreach, chame context.SemanticModel.GetTypeInfo() para recuperar
informações detalhadas sobre o tipo declarado da declaração local:
Em seguida, dentro do loop foreach , verifique cada inicializador para garantir que ele pode ser convertido no tipo
de variável. Adicione a seguinte verificação depois de garantir que o inicializador é uma constante:
// Ensure that the initializer value can be converted to the type of the
// local declaration without a user-defined conversion.
var conversion = context.SemanticModel.ClassifyConversion(initializer.Value, variableType);
if (!conversion.Exists || conversion.IsUserDefined)
{
return;
}
A próxima alteração é realizada com base na última. Antes da chave de fechamento do primeiro loop foreach,
adicione o código a seguir para verificar o tipo da declaração de local quando a constante é uma cadeia de
caracteres ou valor nulo.
// Special cases:
// * If the constant value is a string, the type of the local declaration
// must be System.String.
// * If the constant value is null, the type of the local declaration must
// be a reference type.
if (constantValue.Value is string)
{
if (variableType.SpecialType != SpecialType.System_String)
{
return;
}
}
else if (variableType.IsReferenceType && constantValue.Value != null)
{
return;
}
Você precisa escrever um pouco mais de código no seu provedor de correção de código para substituir a palavra-
chave var' com o nome do tipo correto. Retorne ao CodeFixProvider.cs. O código que você adicionará realizará as
seguintes etapas:
Verifique se a declaração é uma declaração var e se afirmativo:
Crie um novo tipo para o tipo inferido.
Certifique-se de que a declaração de tipo não é um alias. Em caso afirmativo, é legal declarar const var .
Certifique-se de que var não é um nome de tipo neste programa. (Em caso afirmativo, const var é legal).
Simplificar o nome completo do tipo
Isso soa como muito código. Mas não é. Substitua a linha que declara e inicializa newLocal com o código a seguir.
Ele é colocado imediatamente após a inicialização de newModifiers :
// If the type of the declaration is 'var', create a new type name
// for the inferred type.
var variableDeclaration = localDeclaration.Declaration;
var variableTypeName = variableDeclaration.Type;
if (variableTypeName.IsVar)
{
var semanticModel = await document.GetSemanticModelAsync(cancellationToken);
// Special case: Ensure that 'var' isn't actually an alias to another type
// (e.g. using var = System.String).
var aliasInfo = semanticModel.GetAliasInfo(variableTypeName);
if (aliasInfo == null)
{
// Retrieve the type inferred for var.
var type = semanticModel.GetTypeInfo(variableTypeName).ConvertedType;
// Special case: Ensure that 'var' isn't actually a type named 'var'.
if (type.Name != "var")
{
// Create a new TypeSyntax for the inferred type. Be careful
// to keep any leading and trailing trivia from the var keyword.
var typeName = SyntaxFactory.ParseTypeName(type.ToDisplayString())
.WithLeadingTrivia(variableTypeName.GetLeadingTrivia())
.WithTrailingTrivia(variableTypeName.GetTrailingTrivia());
Você precisará adicionar uma instrução using para usar o tipo Simplifier:
using Microsoft.CodeAnalysis.Simplification;
Execute seus testes, que devem todos ser aprovados. Dê parabéns a si mesmo, executando seu analisador
concluído. Pressione Ctrl + F5 para executar o projeto do analisador em uma segunda instância do Visual Studio
com a extensão de versão prévia do Roslyn carregada.
Na segunda instância do Visual Studio, crie um novo projeto de Aplicativo de Console de C# e adicione
int x = "abc"; ao método Main. Graças à primeira correção de bug, nenhum aviso deve ser relatado para esta
declaração de variável local (embora haja um erro do compilador, conforme esperado).
Em seguida, adicione object s = "abc"; ao método Main. Devido à segunda correção de bug, nenhum aviso
deve ser relatado.
Por fim, adicione outra variável local que usa a palavra-chave var . Você verá que um aviso é relatado e uma
sugestão é exibida abaixo e a esquerda.
Mova o cursor do editor sobre o sublinhado ondulado e pressione Ctrl +. para exibir a correção de código
sugerida. Ao selecionar a correção de código, observe que a palavra-chave var' agora é manipulada
corretamente.
Por fim, adicione o seguinte código:
int i = 2;
int j = 32;
int k = i + j;
Após essas alterações, você obtém linhas onduladas vermelhas apenas nas duas primeiras variáveis. Adicione
const para ambos i e j , e você receberá um novo aviso em k porque ele agora pode ser const .
Parabéns! Você criou sua primeira extensão do .NET Compiler Platform que executa análise de código com o
sistema em funcionamento para detectar um problema e fornece uma correção rápida para corrigi-lo. Ao longo do
caminho, você aprendeu muitas das APIs de código que fazem parte do SDK do .NET Compiler Platform (APIs do
Roslyn). Você pode verificar seu trabalho comparando-o à amostra concluída em nosso repositório GitHub de
exemplos. Ou você pode baixar o arquivo zip do projeto concluído
Outros recursos
Introdução à análise de sintaxe
Introdução à análise semântica
Guia de programação em C#
27/11/2019 • 2 minutes to read • Edit Online
Seções de programa
Por dentro de um programa em C#
Main() e argumentos de linha de comando
Seções da linguagem
Instruções, expressões e operadores
Tipos
Classes e Structs
Interfaces
Tipos de enumeração
Delegados
Matrizes
Cadeias de Caracteres
Propriedades
Indexadores
Eventos
Genéricos
Iteradores
Expressões de consulta LINQ
Expressões Lambda
Namespaces
Código não seguro e ponteiros
Comentários da documentação XML
Seções da plataforma
Domínios do aplicativo
Assemblies no .NET
Atributos
Coleções
Exceções e manipulação de exceções
Sistema de arquivos e o Registro (Guia de programação em C#)
Interoperabilidade
Reflexão
Consulte também
Referência de C#
Por dentro de um programa em C#
23/10/2019 • 2 minutes to read • Edit Online
A seção discute a estrutura geral de um programa em C# e inclui o exemplo padrão "Olá, mundo!" .
Nesta seção
Hello World – seu primeiro programa
Estrutura geral de um programa em C#
Nomes de identificadores
Convenções de codificação em C#
Seções relacionadas
Introdução ao C#
Guia de Programação em C#
Referência de C#
Exemplos e tutoriais
Especificação da linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a fonte
definitiva para a sintaxe e o uso de C#.
Consulte também
Guia de Programação em C#
Olá, Mundo--seu primeiro programa
31/10/2019 • 7 minutes to read • Edit Online
Neste artigo, você usará o Visual Studio para criar o "Olá, Mundo!" tradicional programa. O Visual Studio é um
IDE (ambiente de desenvolvimento integrado) profissional com muitos recursos projetados para o
desenvolvimento do .NET. Você usará apenas alguns dos recursos do Visual Studio para criar esse programa. Para
saber mais sobre o Visual Studio, consulte introdução com C#o Visual .
NOTE
Seu computador pode mostrar diferentes nomes ou locais para alguns dos elementos de interface do usuário do Visual
Studio nas instruções a seguir. A edição do Visual Studio que você possui e as configurações que você usa determinam esses
elementos. Para obter mais informações, consulte Personalizando o IDE.
Selecione criar um novo projeto no canto inferior direito da imagem. O Visual Studio exibe a caixa de diálogo
novo projeto :
NOTE
Se esta for a primeira vez que você iniciou o Visual Studio, a lista de modelos de projetos recentes estará vazia.
Na caixa de diálogo novo projeto, escolha "aplicativo de console (.NET Core)" e, em seguida, pressione Avançar.
Dê um nome ao seu projeto, como "HelloWorld", depois pressione criar.
O Visual Studio abre seu projeto. Já é um "Olá, Mundo!" básico . Pressione Ctrl + F5 para executar o projeto. O
Visual Studio cria seu projeto, convertendo o código-fonte em um executável. Em seguida, ele inicia uma janela de
comando que executa o novo aplicativo. Você deve ver o seguinte texto na janela:
Hello World!
Elementos de um C# programa
Vamos examinar as partes importantes deste programa. A primeira linha contém um comentário. Os caracteres
// convertem o restante da linha em um comentário.
Você também pode comentar um bloco de texto, colocando-o entre os caracteres /* e */ . Isso é mostrado no
exemplo a seguir.
/* A "Hello World!" program in C#.
This program displays the string "Hello World!" on the screen. */
Um aplicativo de console do C# deve conter um método Main , no qual o controle começa e termina. O método
Main é o local em que você cria objetos e executa outros métodos.
O método Main é um método estático que reside dentro de uma classe ou um struct. No exemplo de "Hello
World!" anterior, ele reside em uma classe chamada Hello . Você pode declarar o método Main de uma das
seguintes maneiras:
Ele pode retornar void . Isso significa que o programa não retorna um valor.
Ele também pode retornar um inteiro. O inteiro é o código de saída para seu aplicativo.
- ou -
O parâmetro do método Main , args , é um matriz string que contém os argumentos de linha de comando
usados para invocar o programa.
Para obter mais informações sobre como usar argumentos de linha de comando, consulte os exemplos nos
argumentos principal () e de linha de comando.
Entrada e saída
Os programas em C# geralmente usam os serviços de entrada/saída fornecidos pela biblioteca em tempo de
execução do .NET Framework. A instrução System.Console.WriteLine("Hello World!"); usa o método WriteLine.
Esse é um dos métodos de saída da classe Console na biblioteca em tempo de execução. Ele exibe seu parâmetro
de cadeia de caracteres no fluxo de saída padrão, seguido por uma nova linha. Outros métodos Console estão
disponíveis para operações de saída e entrada diferentes. Se você incluir a diretiva using System; no início do
programa, poderá usar diretamente as classes e métodos System sem qualificá-los completamente. Por exemplo,
você pode chamar Console.WriteLine em vez de System.Console.WriteLine :
using System;
Console.WriteLine("Hello World!");
Consulte também
Guia de Programação em C#
Exemplos e tutoriais
Main() e argumentos de linha de comando
Introdução com VisualC#
Estrutura geral de um programa em C# (Guia de
Programação em C#)
04/11/2019 • 2 minutes to read • Edit Online
Os programas C# podem consistir em um ou mais arquivos. Cada arquivo pode conter zero ou mais namespaces.
Um namespace pode conter tipos como classes, estruturas, interfaces, enumerações e delegados, além de outros
namespaces. A seguir está o esqueleto de um programa em C# que contém todos esses elementos.
// A skeleton of a C# program
using System;
namespace YourNamespace
{
class YourClass
{
}
struct YourStruct
{
}
interface IYourInterface
{
}
enum YourEnum
{
}
namespace YourNestedNamespace
{
struct YourStruct
{
}
}
class YourMainClass
{
static void Main(string[] args)
{
//Your program starts here...
}
}
}
Seções relacionadas
Para saber mais:
Classes
Structs
Namespaces
Interfaces
Delegados
Especificação da Linguagem C#
Para obter mais informações, veja Noções básicas na Especificação da linguagem C#. A especificação da
linguagem é a fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Guia de Programação em C#
Por dentro de um programa em C#
Referência de C#
Nomes de identificadores
23/10/2019 • 2 minutes to read • Edit Online
Um identificador é o nome que você atribui a um tipo (classe, interface, struct, delegado ou enumerado),
membro, variável ou namespace. Os identificadores válidos devem seguir estas regras:
Os identificadores devem começar com uma letra ou _ .
Os identificadores podem conter caracteres de letra Unicode, caracteres de dígito decimal, caracteres de
conexão Unicode, caracteres de combinação Unicode ou caracteres de formatação Unicode. Para obter mais
informações sobre as categorias Unicode, consulte o Banco de dados da categoria Unicode. É possível
declarar identificadores que correspondem às palavras-chave em C# usando o prefixo @ no identificador.
O @ não faz parte do nome do identificador. Por exemplo, @if declara um identificador chamado if .
Esses identificadores textuais são destinados principalmente para interoperabilidade com os identificadores
declarados em outras linguagens.
Para obter uma definição completa de identificadores válidos, consulte o tópico de identificadores na
especificação da linguagem C#.
Convenções de nomenclatura
Além das regras, há inúmeras convenções de nomenclatura de identificador usadas em toda as APIs do .NET.
Por convenção, os programas C# usam PascalCase para nomes de tipo, namespaces e todos os membros
públicos. Além disso, as seguintes convenções são comuns:
Os nomes de interface começam com I maiúsculo.
Os tipos de atributo terminam com a palavra Attribute .
Os tipos enumerados usam um substantivo singular para não sinalizadores e um substantivo plural para
sinalizadores.
Os identificadores não devem conter dois caracteres _ consecutivos. Esses nomes estão reservados para
identificadores gerados por compilador.
Especificação da Linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a
fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Guia de Programação em C#
Por dentro de um programa em C#
Referência de C#
Classes
Estruturas
Namespaces
Interfaces
Delegados
Convenções de codificação em C# (Guia de
Programação em C#)
04/11/2019 • 13 minutes to read • Edit Online
Convenções de nomenclatura
Em exemplos curtos que não incluem diretivas using, use as qualificações do namespace. Se você souber
que um namespace é importado por padrão em um projeto, não precisará qualificar totalmente os nomes
desse namespace. Nomes qualificados podem ser interrompidos após um ponto (.) se forem muito longos
para uma única linha, conforme mostrado no exemplo a seguir.
Não é necessário alterar os nomes de objetos que foram criados usando as ferramentas de designer do
Visual Studio para adequá-los a outras diretrizes.
Convenções de Layout
Um bom layout usa formatação para enfatizar a estrutura do código e para facilitar a leitura de código. Exemplos e
amostras Microsoft estão em conformidade com as seguintes convenções:
Use as configurações padrão do Editor de códigos (recuo inteligente, recuos de quatro caracteres, guias
salvas como espaços). Para obter mais informações, consulte Opções, Editor de Texto, C#, Formatação.
Gravar apenas uma instrução por linha.
Gravar apenas uma declaração por linha.
Se as linhas de continuação não devem recuar automaticamente, recue-as uma tabulação (quatro espaços).
Adicione pelo menos uma linha em branco entre as definições de método e de propriedade.
Use parênteses para criar cláusulas em uma expressão aparente, conforme mostrado no código a seguir.
if ((val1 > val2) && (val1 > val3))
{
// Take appropriate action.
}
Comentando Convenções
Coloque o comentário em uma linha separada, não no final de uma linha de código.
Comece o texto do comentário com uma letra maiúscula.
Termine o texto do comentário com um ponto final.
Insira um espaço entre o delimitador de comentário (/ /) e o texto do comentário, conforme mostrado no
exemplo a seguir.
Diretrizes de Linguagem
As seções a seguir descrevem práticas que a equipe de C# segue para preparar exemplos e amostras do código.
Tipo de dados da cadeia de caracteres
Use a interpolação de cadeia de caracteres para concatenar cadeias de caracteres curtas, como é mostrado
no código a seguir.
Para acrescentar cadeias de caracteres em loops, especialmente quando você estiver trabalhando com
grandes quantidades de texto, use um objeto StringBuilder.
// When the type of a variable is clear from the context, use var
// in the declaration.
var var1 = "This is clearly a string.";
var var2 = 27;
var var3 = Convert.ToInt32(Console.ReadLine());
Não use var quando o tipo não estiver aparente no lado direito da atribuição.
// When the type of a variable is not clear from the context, use an
// explicit type.
int var4 = ExampleClass.ResultSoFar();
Não se baseie no nome da variável para especificar o tipo dela. Ele pode não estar correto.
// Preferred syntax. Note that you cannot use var here instead of string[].
string[] vowels1 = { "a", "e", "i", "o", "u" };
// If you specify an array size, you must initialize the elements one at a time.
var vowels3 = new string[5];
vowels3[0] = "a";
vowels3[1] = "e";
// And so on.
Delegados
Use a sintaxe concisa ao criar instâncias de um tipo delegado.
// First, in class Program, define the delegate type and a method that
// has a matching signature.
Simplifique o código usando a instrução using do #C. Se você tiver uma instrução try-finally na qual o único
código do bloco finally é uma chamada para o método Dispose, use, em vez disso, uma instrução using .
// This try-finally statement only calls Dispose in the finally block.
Font font1 = new Font("Arial", 10.0f);
try
{
byte charset = font1.GdiCharSet;
}
finally
{
if (font1 != null)
{
((IDisposable)font1).Dispose();
}
}
Operadores && e ||
Para evitar exceções e aumentar o desempenho ignorando comparações desnecessárias, use && em vez de
& e || em vez de | ao executar comparações, conforme mostrado no exemplo a seguir.
Operador New
Use um formulário conciso de instanciação de objeto com digitação implícita, conforme mostrado na
declaração a seguir.
Tratamento de Evento
Se você estiver definindo um manipulador de eventos que não necessita ser removido posteriormente, use
uma expressão lambda.
public Form2()
{
// You can use a lambda expression to define an event handler.
this.Click += (s, e) =>
{
MessageBox.Show(
((MouseEventArgs)e).Location.ToString());
};
}
Membros Estáticos
Chame membros estáticos usando o nome de classe: ClassName.StaticMember. Essa prática torna o código
mais legível, tornando o acesso estático limpo. Não qualifique um membro estático definido em uma classe
base com o nome de uma classe derivada. Enquanto esse código é compilado, a leitura do código fica incorreta
e o código poderá ser danificado no futuro se você adicionar um membro estático com o mesmo nome da
classe derivada.
Consultas LINQ
Use nomes significativos para variáveis de consulta. O exemplo a seguir usa seattleCustomers para os
clientes que estão localizados em Seattle.
Use aliases para se certificar de que os nomes de propriedades de tipos anônimos sejam colocados
corretamente em maiúsculas, usando o padrão Pascal-Case.
var localDistributors =
from customer in customers
join distributor in distributors on customer.City equals distributor.City
select new { Customer = customer, Distributor = distributor };
Renomeie propriedades quando os nomes de propriedades no resultado forem ambíguos. Por exemplo, se
a sua consulta retornar um nome de cliente e uma ID de distribuidor, em vez de deixá-los como Name e ID
no resultado, renomeie-os para esclarecer que Name é o nome de um cliente, e ID é a identificação de um
distribuidor.
var localDistributors2 =
from customer in customers
join distributor in distributors on customer.City equals distributor.City
select new { CustomerName = customer.Name, DistributorID = distributor.ID };
Alinhe cláusulas de consulta na cláusula from, conforme mostrado nos exemplos anteriores.
Use cláusulas where antes de outras cláusulas de consulta para garantir que cláusulas de consulta
posteriores operem no conjunto de dados filtrado e reduzido.
Use várias cláusulas from em vez de uma cláusula join para acessar coleções internas. Por exemplo, cada
coleção de objetos Student pode conter um conjunto de pontuações no teste. Quando a próxima consulta
for executada, ela retorna cada pontuação que seja acima de 90, juntamente com o sobrenome do estudante
que recebeu a pontuação.
// Use a compound from to access the inner sequence within each element.
var scoreQuery = from student in students
from score in student.Scores
where score > 90
select new { Last = student.LastName, score };
Segurança
Siga as diretrizes em Diretrizes de codificação segura.
Consulte também
Convenções de codificação do Visual Basic
Diretrizes de codificação segura
Main() e argumentos de linha de comando (Guia
de Programação em C#)
27/11/2019 • 4 minutes to read • Edit Online
O método Main é o ponto de entrada de um aplicativo C#. (As bibliotecas e os serviços não exigem um
método Main como um ponto de entrada.) Quando o aplicativo é iniciado, o método Main é o primeiro
método que é invocado.
Pode haver apenas um ponto de entrada em um programa C#. Se tiver mais de uma classe que tenha um
Main método, você deverá compilar seu programa com a opção do compilador /main para especificar qual
método Main será usado como ponto de entrada. Para obter mais informações, consulte -MainC# (opções do
compilador).
class TestClass
{
static void Main(string[] args)
{
// Display the number of command line arguments.
Console.WriteLine(args.Length);
}
}
Visão geral
O método Main é o ponto de entrada de um programa executável; é onde o controle do programa começa
e termina.
Main é declarado dentro de uma classe ou struct. O Main deve ser estático e não precisa ser público. ( No
exemplo anterior, ele recebe o acesso padrão de Private.) A classe ou struct delimitador não precisa ser
estático.
O Main pode ter um tipo de retorno void , int ou, a partir do C# 7.1, Task ou Task<int> .
Se e somente se Main retornar um Task ou Task<int> , a declaração de Main pode incluir o modificador
async . Observe que isso exclui especificamente um método async void Main .
O método Main pode ser declarado com ou sem um parâmetro string[] que contém os argumentos de
linha de comando. Ao usar o Visual Studio para criar aplicativos do Windows, você pode adicionar o
parâmetro manualmente ou, caso contrário, usar o método GetCommandLineArgs() para obter os
argumentos de linha de comando. Os parâmetros são lidos como argumentos de linha de comando
indexados por zero. Ao contrário de C++C e, o nome do programa não é tratado como o primeiro
argumento de linha de comando na matriz de args , mas é o primeiro elemento do método
GetCommandLineArgs().
A seguir está uma lista de assinaturas de Main válidas:
public static void Main() { }
public static int Main() { }
public static void Main(string[] args) { }
public static int Main(string[] args) { }
public static async Task Main() { }
public static async Task<int> Main() { }
public static async Task Main(string[] args) { }
public static async Task<int> Main(string[] args) { }
A adição dos tipos de retorno async , Task e Task<int> simplifica o código do programa quando os
aplicativos do console precisam iniciar e realizar operações assíncronas await no Main .
especificação da linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a
fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Build pela linha de comando com csc.exe
Guia de Programação em C#
Métodos
Por dentro de um programa em C#
Argumentos de linha de comando (Guia de
Programação em C#)
30/10/2019 • 5 minutes to read • Edit Online
Você pode enviar argumentos para o método Main definindo o método de uma das seguintes maneiras:
NOTE
Para habilitar argumentos de linha de comando no método Main em um aplicativo Windows Forms, você deve modificar
manualmente a assinatura de Main em Program.cs. O código gerado pelo Designer de Formulários do Windows cria um
Main sem um parâmetro de entrada. Você também pode usar Environment.CommandLine ou
Environment.GetCommandLineArgs para acessar os argumentos de linha de comando de qualquer ponto em um console ou
um aplicativo do Windows.
O parâmetro do método Main é uma matriz String que representa os argumentos de linha de comando.
Geralmente você determina se os argumentos existem testando a propriedade Length , por exemplo:
if (args.Length == 0)
{
System.Console.WriteLine("Please enter a numeric argument.");
return 1;
}
Você também pode converter os argumentos de cadeia de caracteres em tipos numéricos, usando a classe
Convert ou o método Parse . Por exemplo, a instrução a seguir converte o string em um número long usando
o método Parse:
Também é possível usar o tipo long de C#, que funciona como alias de Int64 :
Você também pode usar o método da classe Convert ,o ToInt64 , para fazer a mesma coisa:
Exemplo
O exemplo a seguir mostra como usar argumentos de linha de comando em um aplicativo de console. O
aplicativo recebe um argumento em tempo de execução, converte o argumento em um número inteiro e calcula o
fatorial do número. Se nenhum argumento for fornecido, o aplicativo emitirá uma mensagem que explica o uso
correto do programa.
Para compilar e executar o aplicativo em um prompt de comando, siga estas etapas:
1. Cole o código a seguir em qualquer editor de texto e, em seguida, salve o arquivo como um arquivo de
texto com o nome factorial.cs.
// Add a using directive for System if the directive isn't already present.
class MainClass
{
static int Main(string[] args)
{
// Test if input arguments were supplied.
if (args.Length == 0)
{
Console.WriteLine("Please enter a numeric argument.");
Console.WriteLine("Usage: Factorial <num>");
return 1;
}
// Calculate factorial.
long result = Functions.Factorial(num);
// Print result.
if (result == -1)
Console.WriteLine("Input must be >= 0 and <= 20.");
else
Console.WriteLine($"The Factorial of {num} is {result}.");
return 0;
}
}
// If 3 is entered on command line, the
// output reads: The factorial of 3 is 6.
2. Na tela Inicial ou no menu Iniciar, abra uma janela de Prompt de Comando do Desenvolvedor do
Visual Studio e, em seguida, navegue até a pasta que contém o arquivo que você acabou de criar.
3. Digite o seguinte comando para compilar o aplicativo.
csc Factorial.cs
Se seu aplicativo não tiver erros de compilação, um arquivo executável chamado fatorial. exe será criado.
4. Digite o seguinte comando para calcular o fatorial de 3:
Factorial 3
NOTE
Ao executar um aplicativo no Visual Studio, você pode especificar argumentos de linha de comando na Página de depuração,
Designer de Projeto.
Consulte também
System.Environment
Guia de Programação em C#
Main() e argumentos de linha de comando
Como exibir argumentos de linha de comando
Valores de retorno de Main()
Classes
Como: exibir argumentos de linha de comando (Guia
de Programação em C#)
23/10/2019 • 2 minutes to read • Edit Online
Os argumentos fornecidos a um executável na linha de comando são acessíveis por meio de um parâmetro
opcional para Main . Os argumentos são fornecidos na forma de uma matriz de cadeias de caracteres. Cada
elemento da matriz contém um argumento. O espaço em branco entre os argumentos é removido. Por exemplo,
considere essas invocações de linha de comando de um executável fictício:
executável.exe a b c "a"
"b"
"c"
"dois"
"three"
NOTE
Quando estiver executando um aplicativo no Visual Studio, você pode especificar argumentos de linha de comando na
Página de depuração, Designer de Projeto.
Exemplo
Este exemplo exibe os argumentos de linha de comando passados a um aplicativo de linha de comando. A saída
mostrada é para a primeira entrada da tabela acima.
class CommandLine
{
static void Main(string[] args)
{
// The Length property provides the number of array elements.
Console.WriteLine($"parameter count = {args.Length}");
Consulte também
Guia de Programação em C#
Build pela linha de comando com csc.exe
Main() e argumentos de linha de comando
Valores de retorno de Main()
Valores retornados de Main() (Guia de Programação
em C#)
23/10/2019 • 5 minutes to read • Edit Online
Se o valor retornado de Main não for usado, o retorno de void permite um código um pouco mais simples. No
entanto, o retorno de um inteiro habilita o programa a comunicar informações de status para outros programas
ou scripts, que invocam o arquivo executável. O valor retornado de Main é tratado como o código de saída para o
processo. Se void for retornado de Main , o código de saída será implicitamente 0 . O exemplo a seguir mostra
como o valor retornado de Main pode ser acessado.
Exemplo
Este exemplo usa ferramentas de linha de comando do .NET Core. Se você não estiver familiarizado com as
ferramentas de linha de comando do .NET Core, poderá aprender sobre elas neste Tópico de introdução.
Modifique o método Main em program.cs da seguinte maneira:
Quando um programa é executado no Windows, qualquer valor retornado da função Main é armazenado em
uma variável de ambiente. Essa variável de ambiente pode ser recuperada usando ERRORLEVEL de um arquivo em
lotes ou $LastExitCode do PowerShell.
Você pode criar o aplicativo usando o comando dotnet build da CLI do dotnet.
Em seguida, crie um script do Powershell para executar o aplicativo e exibir o resultado. Cole o código a seguir em
um arquivo de texto e salve-o como test.ps1 na pasta que contém o projeto. Execute o script do PowerShell
digitando test.ps1 no prompt do PowerShell.
Como o código retorna zero, o arquivo em lotes relatará êxito. No entanto, se você alterar o MainReturnValTest.cs
para retornar um valor diferente de zero e recompilar o programa, a execução subsequente do script do
PowerShell reportará falha.
dotnet run
if ($LastExitCode -eq 0) {
Write-Host "Execution succeeded"
} else
{
Write-Host "Execution Failed"
}
Write-Host "Return value = " $LastExitCode
Saída de exemplo
Execution succeeded
Return value = 0
NOTE
Se os exemplos usassem o modificador async no método Main , o compilador geraria o mesmo código.
Consulte também
Guia de Programação em C#
Referência de C#
Main() e argumentos de linha de comando
Como: exibir argumentos de linha de comando
Conceitos de programação (C#)
23/10/2019 • 2 minutes to read • Edit Online
Nesta seção
TÍTULO DESCRIÇÃO
Programação assíncrona com async e await (C#) Descreve como criar soluções assíncronas usando as palavras-
chave async e await no C#. Inclui um passo a passo.
Coleções (C#) Descreve alguns dos tipos de coleções fornecidos pelo .NET
Framework. Demonstra como usar coleções simples e coleções
de pares chave/valor.
Árvores de expressão (C#) Explica como você pode usar árvores de expressão para
habilitar a modificação dinâmica de código executável.
LINQ (consulta integrada à linguagem) (C#) Discute os recursos avançados de consulta na sintaxe de
linguagem do C# e o modelo para consultar bancos de dados
relacionais, documentos XML, conjuntos de dados e coleções
na memória.
Programação orientada a objeto (C#) Descreve os conceitos comuns orientados a objetos, incluindo
encapsulamento, herança e polimorfismo.
Reflexão (C#) Explica como usar a reflexão para criar dinamicamente uma
instância de um tipo, associar o tipo a um objeto existente ou
obter o tipo de um objeto existente e invocar seus métodos
ou acessar suas propriedades e campos.
Dicas de desempenho Discute várias regras básicas que podem ajudá-lo a aumentar
o desempenho do seu aplicativo.
Programação assíncrona com async e await
29/11/2019 • 18 minutes to read • Edit Online
O modelo TAP (programação assíncrona Task) proporciona uma abstração em código assíncrono. Você
escreve o código como uma sequência de instruções, como usual. Você pode ler o código como se cada
instrução fosse concluída antes do início da próxima. O compilador realiza uma série de transformações,
porque algumas dessas instruções podem iniciar o trabalho e retornar um Task que representa o trabalho em
andamento.
Essa é a meta dessa sintaxe: habilitar um código que leia como uma sequência de instruções, mas que execute
em uma ordem muito mais complicada com base na alocação de recurso externo e em quando as tarefas são
concluídas. Isso é semelhante à maneira como as pessoas dão instruções para processos que incluem tarefas
assíncronas. Neste artigo, você usará um exemplo com instruções para fazer um café da manhã e ver como as
palavras-chave async e await facilitam raciocinar sobre o código que inclui uma série de instruções
assíncronas. Você deve escrever as instruções de maneira parecida com a lista a seguir para explicar como
fazer um café da manhã:
1. Encher uma xícara de café.
2. Aquecer uma frigideira e, em seguida, fritar dois ovos.
3. Frita três fatias de bacon.
4. Torrar dois pedaços de pão.
5. Adicionar manteiga e a geleia na torrada.
6. Encher um copo com suco de laranja.
Se tivesse experiência em culinária, você executaria essas instruções assincronamente. Você iniciaria
aquecendo a frigideira para os ovos e, em seguida, começaria a preparar o bacon. Você colocaria o pão na
torradeira e começaria a preparar os ovos. Em cada etapa do processo, iniciaria uma tarefa e voltaria sua
atenção para as tarefas que estivessem prontas para a sua atenção.
Preparar o café da manhã é um bom exemplo de trabalho assíncrono que não é paralelo. Uma pessoa (ou um
thread) pode lidar com todas essas tarefas. Continuando com a analogia do café da manhã, uma pessoa pode
fazer café da manhã assincronamente iniciando a tarefa seguinte antes de concluir a primeira. O preparo
progride independentemente de haver alguém observando. Assim que inicia o aquecimento da frigideira para
os ovos, você pode começar a fritar o bacon. Quando começar a preparar o bacon, você pode colocar o pão na
torradeira.
Para um algoritmo paralelo, você precisaria de vários cozinheiros (ou threads). Um prepararia os ovos, outro
o bacon e assim por diante. Cada um se concentraria apenas naquela tarefa específica. Cada cozinheiro (ou
thread) ficaria bloqueado de forma síncrona, esperando que o bacon estivesse pronto para ser virado ou que
a torrada pulasse.
Agora, considere essas mesmas instruções escritas como instruções em C#:
static void Main(string[] args)
{
Coffee cup = PourCoffee();
Console.WriteLine("coffee is ready");
Egg eggs = FryEggs(2);
Console.WriteLine("eggs are ready");
Bacon bacon = FryBacon(3);
Console.WriteLine("bacon is ready");
Toast toast = ToastBread(2);
ApplyButter(toast);
ApplyJam(toast);
Console.WriteLine("toast is ready");
Juice oj = PourOJ();
Console.WriteLine("oj is ready");
Console.WriteLine("Breakfast is ready!");
}
Os computadores não interpretam essas instruções da mesma forma que as pessoas. O computador ficará
bloqueado em cada instrução até que o trabalho seja concluído, antes de passar para a próxima instrução. Isso
cria um café da manhã insatisfatório. As tarefas posteriores não seriam iniciadas até que as tarefas anteriores
fossem concluídas. Levaria muito mais tempo para criar o café da manhã e alguns itens ficariam frios antes de
serem servidos.
Se você quiser que o computador execute as instruções acima de forma assíncrona, deverá escrever o código
assíncrono.
Essas questões são importantes para os programas que você escreve atualmente. Ao escrever programas de
cliente, você quer que a interface do usuário responda de acordo com as solicitações do usuário. Seu
aplicativo não deve fazer um telefone parecer travado enquanto ele está baixando dados da Web. Ao escrever
programas de servidor, você não quer threads bloqueados. Esses threads poderiam servir a outras
solicitações. O uso de código síncrono quando existem alternativas assíncronas afeta sua capacidade de
aumentar de forma menos custosa. Você paga pelos threads bloqueados.
Aplicativos modernos bem-sucedidos exigem código assíncrono. Sem suporte de linguagem, escrever código
assíncrono exigia retornos de chamada, eventos de conclusão ou outros meios que obscureciam a intenção
original do código. A vantagem do código síncrono é que ele é fácil de entender. As ações passo a passo
facilitam o exame e o entendimento. Modelos assíncronos tradicionais forçavam você a se concentrar na
natureza assíncrona do código e não nas ações fundamentais do código.
Console.WriteLine("Breakfast is ready!");
}
Esse código não bloqueia enquanto os ovos ou o bacon são preparados. Entretanto, esse código não iniciará
outras tarefas. Você ainda colocaria o pão na torradeira e ficaria olhando até ele pular. Mas, pelo menos, você
responderia a qualquer pessoa que quisesse sua atenção. Em um restaurante em que vários pedidos são
feitos, o cozinheiro pode iniciar o preparo de outro café da manhã enquanto prepara o primeiro.
Agora, o thread trabalhando no café da manhã não fica bloqueado aguardando qualquer tarefa iniciada que
ainda não tenha terminado. Para alguns aplicativos, essa alteração já basta. Um aplicativo de GUI ainda
responde ao usuário com apenas essa alteração. No entanto, neste cenário, você quer mais. Você não deseja
que cada uma das tarefas componentes seja executada em sequência. É melhor iniciar cada uma das tarefas
componentes antes de aguardar a conclusão da tarefa anterior.
Console.WriteLine("Breakfast is ready!");
Em seguida, você pode mover as instruções await do bacon e dos ovos até o final do método, antes de servir
o café da manhã:
Console.WriteLine("Breakfast is ready!");
O código anterior funciona melhor. Você inicia todas as tarefas assíncronas ao mesmo tempo. Você aguarda
cada tarefa somente quando precisar dos resultados. O código anterior pode ser semelhante a um código em
um aplicativo Web que faz solicitações de diferentes microsserviços e combina os resultados em uma única
página. Você fará todas as solicitações imediatamente e, em seguida, await em todas essas tarefas e
comporá a página da Web.
IMPORTANT
A composição de uma operação assíncrona seguida por trabalho síncrono é uma operação assíncrona. Explicando de
outra forma, se qualquer parte de uma operação for assíncrona, toda a operação será assíncrona.
O código anterior mostrou que você pode usar objetos Task ou Task<TResult> para manter tarefas em
execução. Você await em cada tarefa antes de usar seu resultado. A próxima etapa é criar métodos que
declarem a combinação de outro trabalho. Antes de servir o café da manhã, você quer aguardar a tarefa que
representa torrar o pão antes de adicionar manteiga e geleia. Você pode declarar esse trabalho com o código
a seguir:
O método anterior tem o modificador async na sua assinatura. Isso sinaliza ao compilador que esse método
contém uma instrução await ; ele contém operações assíncronas. Este método representa a tarefa que torra o
pão e, em seguida, adiciona manteiga e geleia. Esse método retorna um Task<TResult> que representa a
composição dessas três operações. O principal bloco de código agora se torna:
Console.WriteLine("Breakfast is ready!");
A alteração anterior ilustrou uma técnica importante para trabalhar com código assíncrono. Você pode
compor tarefas, separando as operações em um novo método que retorna uma tarefa. Você pode escolher
quando aguardar essa tarefa. Você pode iniciar outras tarefas simultaneamente.
Outra opção é usar WhenAny, que retorna uma Task<Task> que é concluída quando qualquer um dos
argumentos é concluído. Você pode aguardar a tarefa retornada, sabendo que ela já foi concluída. O código a
seguir mostra como você poderia usar WhenAny para aguardar a primeira tarefa concluir e, em seguida,
processar seu resultado. Depois de processar o resultado da tarefa concluída, você remove essa tarefa
concluída da lista de tarefas passada para WhenAny .
Depois de todas essas alterações, a versão final do Main fica parecida o código a seguir:
static async Task Main(string[] args)
{
Coffee cup = PourCoffee();
Console.WriteLine("coffee is ready");
var eggsTask = FryEggsAsync(2);
var baconTask = FryBaconAsync(3);
var toastTask = MakeToastWithButterAndJamAsync(2);
Esse código final é assíncrono. Ele reflete mais precisamente como uma pessoa poderia preparar um café da
manhã. Compare o código anterior com o primeiro exemplo de código neste artigo. As ações principais
permanecem claras ao ler o código. Você pode ler esse código da mesma forma como faria ao ler essas
instruções para fazer um café da manhã no início deste artigo. Os recursos de linguagem para async e
await fornecem a tradução que todas as pessoas fazem para seguir essas instruções escritas: iniciar tarefas
assim que possível e não ficar bloqueado ao aguardar a conclusão de tarefas.
Modelo de programação assíncrona de tarefa
25/11/2019 • 32 minutes to read • Edit Online
É possível evitar gargalos de desempenho e aprimorar a resposta geral do seu aplicativo usando a programação
assíncrona. No entanto, as técnicas tradicionais para escrever aplicativos assíncronos podem ser complicadas,
dificultando sua escrita, depuração e manutenção.
O C# 5 apresentou uma programação assíncrona de abordagem simplificada que aproveita o suporte assíncrono
no .NET Framework 4.5 e superior, no .NET Core e no Windows Runtime. O compilador faz o trabalho difícil que o
desenvolvedor costumava fazer, e seu aplicativo mantém a estrutura lógica que se assemelha ao código síncrono.
Como resultado, você obtém todas as vantagens da programação assíncrona com uma fração do esforço.
Este tópico oferece uma visão geral de quando e como usar a programação assíncrona e inclui links para tópicos
de suporte que contêm detalhes e exemplos.
A assincronia é especialmente importante para aplicativos que acessam o thread de interface de usuário porque
todas as atividades relacionadas à interface do usuário normalmente compartilham um único thread. Se um
processo for bloqueado em um aplicativo síncrono, todos serão bloqueados. Seu aplicativo para de responder, o
que poderia levar você a concluir que ele falhou quando, na verdade, está apenas aguardando.
Quando você usa métodos assíncronos, o aplicativo continua a responder à interface do usuário. Você poderá
redimensionar ou minimizar uma janela, por exemplo, ou fechar o aplicativo se você não desejar aguardar sua
conclusão.
A abordagem baseada em assincronia adiciona o equivalente de uma transmissão automática à lista de opções
disponíveis para escolha ao criar operações assíncronas. Ou seja, você obtém todos os benefícios da programação
assíncrona tradicional, mas com muito menos esforço do desenvolvedor.
// GetStringAsync returns a Task<string>. That means that when you await the
// task you'll get a string (urlContents).
Task<string> getStringTask = client.GetStringAsync("https://docs.microsoft.com/dotnet");
// You can do work here that doesn't rely on the string from GetStringAsync.
DoIndependentWork();
Você pode aprender várias práticas com a amostra anterior. Comece com a assinatura do método. Ele inclui o
modificador async . O tipo de retorno é Task<int> (confira a seção "Tipos de retorno" para obter mais opções). O
nome do método termina com Async . No corpo do método, GetStringAsync retorna uma Task<string> . Isso
significa que, quando você executar await em uma tarefa, obterá uma string ( urlContents ). Antes de aguardar a
tarefa, você poderá fazer um trabalho que não dependa da string em GetStringAsync .
Preste muita atenção no operador await . Ele suspende AccessTheWebAsync ;
AccessTheWebAsync não poderá continuar enquanto getStringTask não for concluída.
Enquanto isso, o controle é retornado ao chamador de AccessTheWebAsync .
O controle será retomado aqui quando a getStringTask for concluída.
Em seguida, o operador await recupera o resultado string de getStringTask .
A instrução de retorno especifica um resultado inteiro. Os métodos que estão aguardando AccessTheWebAsync
recuperar o valor de comprimento.
Se AccessTheWebAsync não tiver nenhum trabalho que possa fazer entre chamar GetStringAsync e aguardar a
conclusão, você poderá simplificar o código ao chamar e esperar na instrução única a seguir.
Os números no diagrama correspondem às etapas a seguir, iniciadas quando o usuário clica no botão "Iniciar".
1. Um manipulador de eventos chama e aguarda o método Async AccessTheWebAsync .
2. AccessTheWebAsync cria uma instância de HttpClient e chama o método assíncrono GetStringAsync para
baixar o conteúdo de um site como uma cadeia de caracteres.
3. Algo acontece em GetStringAsync que suspende o andamento. Talvez ele deva aguardar o download de um
site ou alguma outra atividade causadora de bloqueio. Para evitar o bloqueio de recursos, GetStringAsync
transfere o controle para seu chamador, AccessTheWebAsync .
GetStringAsync retorna um Task<TResult>, em que TResult é uma cadeia de caracteres, e
AccessTheWebAsync atribui a tarefa à variável getStringTask . A tarefa representa o processo contínuo para a
chamada a GetStringAsync , com um compromisso de produzir um valor de cadeia de caracteres real
quando o trabalho estiver concluído.
4. Como o getStringTask ainda não foi esperado, AccessTheWebAsync pode continuar com outro trabalho que
não depende do resultado final de GetStringAsync . O trabalho é representado por uma chamada ao método
síncrono DoIndependentWork .
5. DoIndependentWork é um método síncrono que faz seu trabalho e retorna ao seu chamador.
6. AccessTheWebAsync está sem trabalho que ele possa executar sem um resultado de getStringTask . Em
seguida, AccessTheWebAsync deseja calcular e retornar o comprimento da cadeia de caracteres baixada, mas
o método não poderá calcular o valor enquanto o método tiver a cadeia de caracteres.
Portanto, AccessTheWebAsync usa um operador await para suspender seu andamento e para transferir o
controle para o método que chamou AccessTheWebAsync . AccessTheWebAsync retorna um Task<int> ao
chamador. A tarefa representa uma promessa de produzir um resultado inteiro que é o comprimento da
cadeia de caracteres baixada.
NOTE
Se GetStringAsync (e, portanto, getStringTask ) for concluído antes que AccessTheWebAsync o aguarde, o
controle permanecerá em AccessTheWebAsync . A despesa de suspender e depois retornar para
AccessTheWebAsync seria desperdiçada caso o processo assíncrono chamado ( getStringTask ) já tivesse sido
concluído e AccessTheWebAsync não tivesse que aguardar o resultado final.
Threads
Os métodos assíncronos destinam-se a ser operações não causadoras de bloqueios. Uma expressão await em um
método assíncrono não bloqueia o thread atual enquanto a tarefa esperada está em execução. Em vez disso, a
expressão anterior assina o restante do método como uma continuação e retorna o controle para o chamador do
método assíncrono.
As palavras-chave async e await não fazem com que threads adicionais sejam criados. Os métodos assíncronos
não exigem multithreading, pois um método assíncrono não executa em seu próprio thread. O método é executado
no contexto de sincronização atual e usa tempo no thread somente quando o método está ativo. É possível usar
Task.Run para mover o trabalho de CPU associado a um thread em segundo plano, mas um thread em segundo
plano não ajuda com um processo que está apenas aguardando que os resultados tornem-se disponíveis.
A abordagem baseada em async para a programação assíncrona é preferível às abordagens existentes em quase
todos os casos. Essa abordagem é especialmente mais eficiente do que a classe BackgroundWorker para operações
de entrada e saída, porque o código é mais simples e você não precisa se proteger contra condições de corrida. Em
combinação com o método Task.Run, a programação assíncrona é melhor que BackgroundWorker para operações
associadas à CPU, porque a programação assíncrona separa os detalhes de coordenação da execução do código do
trabalho que Task.Run transfere ao pool de threads.
async e await
Se você especificar que um método é assíncrono usando um modificador async, você habilitará os dois recursos a
seguir.
O método assíncrono marcado pode usar await para designar pontos de suspensão. O operador await
informa ao compilador que o método assíncrono não poderá continuar além daquele ponto até que o
processo assíncrono aguardado seja concluído. Enquanto isso, o controle retorna para o chamador do
método assíncrono.
A suspensão de um método assíncrono em uma expressão await não constitui uma saída do método e os
blocos finally não são executados.
O método assíncrono marcado pode ele próprio ser aguardado por métodos que o chamam.
Um método assíncrono normalmente contém uma ou mais ocorrências de um operador await , mas a ausência de
expressões await não causa um erro de compilador. Se um método assíncrono não usar um operador await para
marcar um ponto de suspensão, o método será executado como um método síncrono, apesar do modificador
async . O compilador emite um aviso para esses métodos.
async e await são palavras-chave contextuais. Para obter mais informações e exemplos, consulte os seguintes
tópicos:
async
await
// Calls to GetTaskOfTResultAsync
Task<int> returnedTaskTResult = GetTaskOfTResultAsync();
int intResult = await returnedTaskTResult;
// or, in a single statement
int intResult = await GetTaskOfTResultAsync();
// Calls to GetTaskAsync
Task returnedTask = GetTaskAsync();
await returnedTask;
// or, in a single statement
await GetTaskAsync();
Cada tarefa retornada representa um trabalho em andamento. Uma tarefa encapsula informações sobre o estado
do processo assíncrono e, consequentemente, o resultado final do processo ou a exceção que o processo apresenta
quando não é bem-sucedido.
Um método assíncrono também pode ter um tipo de retorno void . Esse tipo de retorno é usado principalmente
para definir manipuladores de eventos, onde o tipo de retorno void é necessário. Os manipuladores de eventos
assíncronos geralmente servem como o ponto de partida para programas assíncronos.
Um método assíncrono que tem um tipo de retorno void não pode ser aguardado, e o chamador de um método
de retorno nulo não pode capturar nenhuma exceção que o método gera.
O método não pode declarar nenhum parâmetro in, ref ou out, mas pode chamar métodos com tais parâmetros.
Da mesma forma, um método assíncrono não pode retornar um valor por referência, embora possa chamar
métodos com valores retornados ref.
Para obter mais informações e exemplos, consulte Tipos de retorno assíncronos (C#). Para obter mais informações
sobre como capturar exceções nos métodos assíncronos, consulte try-catch.
As APIs assíncronas na programação do Windows Runtime têm um dos seguintes tipos de retorno, que são
semelhantes às tarefas:
IAsyncOperation<TResult>, que corresponde a Task<TResult>
IAsyncAction, que corresponde a Task
IAsyncActionWithProgress<TProgress>
IAsyncOperationWithProgress<TResult, TProgress>
Convenção de nomenclatura
Por convenção, os métodos que geralmente retornam tipos awaitable (por exemplo, Task , Task<T> , ValueTask e
ValueTask<T> ) devem ter nomes que terminam com "Async". Os métodos que iniciam uma operação assíncrona,
mas não retornam um tipo aguardável não devem ter nomes que terminam com "Async", mas podem começar
com "Begin", "Start" ou algum outro verbo que indique que esse método não retorna nem gera o resultado da
operação.
É possível ignorar a convenção quando um evento, uma classe base ou um contrato de interface sugere um nome
diferente. Por exemplo, você não deve renomear manipuladores de eventos comuns, como Button1_Click .
Passo a passo: acessando a Web e Mostra como converter uma solução Exemplo de assincronia: acessando o
usando async e await (C#) síncrona do WPF em uma solução passo a passo da Web
assíncrona do WPF. O aplicativo baixa
uma série de sites.
Como fazer várias solicitações da Web Demonstra como iniciar várias tarefas Exemplo de assincronia: fazer várias
em paralelo usando Async e Await (C#) ao mesmo tempo. solicitações da Web paralelamente
Fluxo de controle em programas Rastreia em detalhes o fluxo de controle Exemplo de assincronia: fluxo de
assíncronos (C#) por meio de uma sucessão de controle em programas assíncronos
expressões de espera em um programa
assíncrono.
TÍTULO DESCRIÇÃO AMOSTRA
Ajuste fino de seu aplicativo assíncrono Mostra como adicionar a seguinte Exemplo de assincronia: ajuste fino de
(C#) funcionalidade à sua solução assíncrona: seu aplicativo
WhenAny: ponte entre o .NET Mostra como criar uma ponte entre Exemplo de assincronia: ponte entre o
Framework e o Windows Runtime tipos Task no .NET Framework e .NET e o Windows Runtime (AsTask e
IAsyncOperations no Windows Runtime WhenAny)
para que você possa usar WhenAny
com um método do Windows Runtime.
Cancelamento assíncrono: ponte entre Mostra como criar uma ponte entre Exemplo de assincronia: ponte entre o
o .NET Framework e o Windows tipos Task no .NET Framework e .NET e o Windows Runtime (AsTask e
Runtime IAsyncOperations no Windows Runtime Cancellation)
para que você possa usar
CancellationTokenSource com um
método do Windows Runtime.
Exemplo completo
O código a seguir é o arquivo MainWindow.XAML.cs do aplicativo WPF que este artigo discute. É possível baixar o
exemplo de Exemplo de assincronia: exemplo de "Programação assíncrona com Async e Await”.
using System;
using System.Threading.Tasks;
using System.Windows;
namespace AsyncFirstExample
{
public partial class MainWindow : Window
{
// Mark the event handler with async so you can use await in it.
private async void StartButton_Click(object sender, RoutedEventArgs e)
{
// Call and await separately.
//Task<int> getLengthTask = AccessTheWebAsync();
//// You can do independent work here.
//int contentLength = await getLengthTask;
resultsTextBox.Text +=
$"\r\nLength of the downloaded string: {contentLength}.\r\n";
}
// GetStringAsync returns a Task<string>. That means that when you await the
// task you'll get a string (urlContents).
Task<string> getStringTask = client.GetStringAsync("https://docs.microsoft.com/dotnet");
// You can do work here that doesn't rely on the string from GetStringAsync.
DoIndependentWork();
void DoIndependentWork()
{
resultsTextBox.Text += "\r\nWorking . . . . . . .\r\n";
}
}
}
// Sample Output:
// Working . . . . . . .
Consulte também
async
await
Programação assíncrona
Visão geral da assincronia
Passo a passo: acessando a Web usando async e
await (C#)
25/11/2019 • 28 minutes to read • Edit Online
É possível escrever programas assíncronos de forma mais fácil e intuitiva usando funcionalidades async/await.
Você pode escrever código assíncrono que se parece com código síncrono e deixar que o compilador trate das
complicadas continuações e funções de retorno de chamada que um código assíncrono normalmente envolve.
Para obter mais informações sobre o recurso Assíncrono, consulte Programação assíncrona com async e await
(C#).
Este passo a passo começa com um aplicativo WPF (Windows Presentation Foundation) síncrono que soma o
número de bytes em uma lista de sites. Em seguida, converte o aplicativo em uma solução assíncrona usando
os novos recursos.
Se não quiser compilar os aplicativos, você poderá baixar o Exemplo de assincronia: acessando o passo a passo
da Web (C# e Visual Basic).
NOTE
Para executar os exemplos, você precisa ter o Visual Studio 2012 ou uma versão mais recente e o .NET Framework 4.5 ou
posterior instalados em seu computador.
using System.Net.Http;
using System.Net;
using System.IO;
resultsTextBox.Clear();
SumPageSizes();
resultsTextBox.Text += "\r\nControl returned to startButton_Click.";
O código chama o método que aciona o aplicativo, SumPageSizes e exibe uma mensagem quando o
controle retorna para startButton_Click .
3. O código para a solução síncrona contém os quatro métodos a seguir:
SumPageSizes , que obtém uma lista de URLs de página da Web de SetUpURLList e chama
GetURLContents e DisplayResults para processar cada URL.
var total = 0;
foreach (var url in urlList)
{
// GetURLContents returns the contents of url as a byte array.
byte[] urlContents = GetURLContents(url);
DisplayResults(url, urlContents);
msdn.microsoft.com/library/windows/apps/br211380.aspx 383832
msdn.microsoft.com 33964
msdn.microsoft.com/library/hh290136.aspx 225793
msdn.microsoft.com/library/ee256749.aspx 143577
msdn.microsoft.com/library/hh290138.aspx 237372
msdn.microsoft.com/library/hh290140.aspx 128279
msdn.microsoft.com/library/dd470362.aspx 157649
msdn.microsoft.com/library/aa578028.aspx 204457
msdn.microsoft.com/library/ms404677.aspx 176405
msdn.microsoft.com/library/ff730837.aspx 143474
Observe que são necessários alguns segundos para exibir as contagens. Durante esse tempo, o thread da
interface do usuário é bloqueado enquanto espera que os recursos solicitados sejam baixados. Como resultado,
você não pode mover, maximizar, minimizar ou até mesmo fechar a janela de exibição após escolher o botão
Iniciar. Esses esforços falham até que as contagens de bytes comecem a aparecer. Se um site não estiver
respondendo, você não terá nenhuma indicação de qual site falhou. É difícil até mesmo parar de esperar e
fechar o programa.
2. GetResponseAsync retorna um Task<TResult>. Nesse caso, a variável de retorno de tarefa, TResult , tem
o tipo WebResponse. A tarefa é uma promessa de produzir um objeto WebResponse verdadeiro após os
dados solicitados terem sido baixados e a tarefa ter sido executada até a conclusão.
Para recuperar o valor WebResponse da tarefa, aplique um operador await à chamada para
GetResponseAsync , como mostra o código a seguir.
O operador await suspende a execução do método atual, GetURLContents , até que a tarefa aguardada
seja concluída. Enquanto isso, o controle retorna para o chamador do método atual. Neste exemplo, o
método atual é GetURLContents e o chamador é SumPageSizes . Quando a tarefa é concluída, o objeto
WebResponse prometido é produzido como o valor da tarefa aguardada e é atribuído à variável response .
A instrução anterior pode ser separada em duas instruções a seguir para esclarecer o que acontece.
4. Tudo o que resta fazer em GetURLContents é ajustar a assinatura do método. Você pode usar o operador
await apenas em métodos que são marcados com o modificador async. Adicione o modificador para
marcar o método como um método assíncrono, como mostra o código a seguir.
5. O tipo de retorno de um método assíncrono só pode ser Task, Task<TResult> ou void em C#.
Normalmente, um tipo de retorno de void é usado somente em um manipulador de eventos
assíncrono, em que void é necessário. Em outros casos, você usa Task(T) se o método concluído tiver
uma instrução return que retorna um valor do tipo T e usa Task se o método concluído não retornar um
valor significativo. Você pode considerar que o tipo de retorno Task significa "Task(void)".
Para obter mais informações, consulte Tipos de retorno assíncronos (C#).
O método GetURLContents tem uma instrução de retorno e a instrução retorna uma matriz de bytes.
Portanto, o tipo de retorno da versão assíncrona é Task(T), em que T é uma matriz de bytes. Faça as
seguintes alterações na assinatura do método:
Altere o tipo de retorno para Task<byte[]> .
Por convenção, métodos assíncronos têm nomes que terminam em "Async", então renomeie o
método GetURLContentsAsync .
O código a seguir mostra essas alterações.
3. Para evitar inserir novamente a operação por acidente, adicione a seguinte instrução à parte superior de
startButton_Click para desabilitar o botão Iniciar.
// Reenable the button in case you want to run the operation again.
startButton.IsEnabled = true;
Para obter mais informações sobre a reentrada, consulte Tratando a reentrada em aplicativos assíncronos
(C#).
4. Por fim, adicione o modificador async à declaração de modo que o manipulador de eventos pode
esperar SumPagSizesAsync .
Normalmente, os nomes dos manipuladores de eventos não são alterados. O tipo de retorno não é
alterado para Task porque os manipuladores de eventos devem retornar void .
A conversão do projeto de um processamento síncrono em um assíncrono está concluída.
2. Em SumPageSizesAsync, , substitua a chamada para seu método GetURLContentsAsync por uma chamada
para o método HttpClient .
Código de exemplo
O código a seguir contém o exemplo completo da conversão de uma solução síncrona em uma solução
assíncrona usando o método GetURLContentsAsync assíncrono que você escreveu. Observe que ele se
assemelha muito à solução síncrona original.
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Data;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Imaging;
using System.Windows.Navigation;
using System.Windows.Shapes;
// Add the following using directives, and add a reference for System.Net.Http.
using System.Net.Http;
using System.IO;
using System.Net;
namespace AsyncExampleWPF
{
public partial class MainWindow : Window
{
public MainWindow()
{
InitializeComponent();
}
resultsTextBox.Clear();
// Reenable the button in case you want to run the operation again.
startButton.IsEnabled = true;
}
var total = 0;
DisplayResults(url, urlContents);
O código a seguir contém o exemplo completo da solução que usa o método HttpClient , GetByteArrayAsync .
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Data;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Imaging;
using System.Windows.Navigation;
using System.Windows.Shapes;
// Add the following using directives, and add a reference for System.Net.Http.
using System.Net.Http;
using System.IO;
using System.Net;
namespace AsyncExampleWPF
{
public partial class MainWindow : Window
{
public MainWindow()
{
InitializeComponent();
}
// Reenable the button in case you want to run the operation again.
startButton.IsEnabled = true;
}
var total = 0;
// The following two lines can replace the previous assignment statement.
//Task<byte[]> getContentsTask = client.GetByteArrayAsync(url);
//byte[] urlContents = await getContentsTask;
DisplayResults(url, urlContents);
Consulte também
Exemplo de assincronia: acessando o passo a passo da Web (C# e Visual Basic)
async
await
Programação assíncrona com async e await (C#)
Tipos de retorno assíncronos (C#)
TAP (programação assíncrona baseada em tarefas)
Como estender a instrução assíncrona usando Task. WhenAll (C#)
Como fazer várias solicitações da Web em paralelo usando Async e Await (C#)
Como estender a instrução assíncrona usando Task.
WhenAll (C#)
25/11/2019 • 15 minutes to read • Edit Online
Você pode melhorar o desempenho da solução assíncrona em Passo a passo: acessando a Web usando async e
await (C#), usando o método Task.WhenAll. Esse método aguarda de maneira assíncrona várias operações
assíncronas, que são representadas como uma coleção de tarefas.
Você deve ter notado no passo a passo que os sites fazem o download em taxas diferentes. Às vezes, um dos
sites está muito lento e isso atrasa todos os downloads restantes. Ao executar as soluções assíncronas que você
compilou no passo a passo, você poderá finalizar o programa facilmente se não quiser esperar, mas uma opção
melhor seria iniciar todos os downloads ao mesmo tempo e permitir que os downloads mais rápidos continuem,
sem aguardar o que está atrasado.
Você aplica o método Task.WhenAll a uma coleção de tarefas. A aplicação de WhenAll retorna uma única tarefa
que não será concluída até a conclusão de cada tarefa na coleção. As tarefas parecem ser executadas em paralelo,
mas não são criados threads adicionais. As tarefas podem ser concluídas em qualquer ordem.
IMPORTANT
Os procedimentos a seguir descrevem as extensões para os aplicativos assíncronos que são desenvolvidos no Passo a
passo: acessando a Web usando async e await (C#). Você pode desenvolver os aplicativos concluindo o passo a passo ou
baixando o código em Exemplos de código para desenvolvedores.
Para executar o exemplo, você deve ter o Visual Studio 2012 ou mais recente instalado no seu computador.
// DisplayResults(url, urlContents);
3. Crie uma coleção de tarefas. O código a seguir define uma consulta que, quando executada pelo método
ToArray, cria uma coleção de tarefas que baixa o conteúdo de cada site. As tarefas são iniciadas quando a
consulta é avaliada.
Adicione o seguinte código ao método SumPageSizesAsync depois da declaração da urlList .
// Create a query.
IEnumerable<Task<int>> downloadTasksQuery =
from url in urlList select ProcessURLAsync(url);
// Use ToArray to execute the query and start the download tasks.
Task<int>[] downloadTasks = downloadTasksQuery.ToArray();
4. Aplique Task.WhenAll à coleção de tarefas downloadTasks . O Task.WhenAll retorna uma única tarefa que
será terminada quando todas as tarefas na coleção de tarefas forem concluídas.
No exemplo a seguir, a expressão await aguarda a conclusão da única tarefa que o WhenAll retorna. A
expressão é avaliada para uma matriz de inteiros, em que cada inteiro é o comprimento de um site
baixado. Adicione o seguinte código ao SumPageSizesAsync , logo após o código que você adicionou na
etapa anterior.
5. Por fim, use o método Sum para calcular a soma dos comprimentos de todos os sites. Adicione a seguinte
linha ao SumPageSizesAsync .
2. Comente ou exclua o For Each ou o loop foreach em SumPageSizesAsync , como mostrado no código a
seguir.
//var total = 0;
//foreach (var url in urlList)
//{
// // GetByteArrayAsync returns a Task<T>. At completion, the task
// // produces a byte array.
// byte[] urlContent = await client.GetByteArrayAsync(url);
// DisplayResults(url, urlContent);
3. Defina uma consulta que, quando executada pelo método ToArray, cria uma coleção de tarefas que baixa
o conteúdo de cada site. As tarefas são iniciadas quando a consulta é avaliada.
Adicione o seguinte código ao método SumPageSizesAsync depois da declaração de client e urlList .
// Create a query.
IEnumerable<Task<int>> downloadTasksQuery =
from url in urlList select ProcessURLAsync(url, client);
// Use ToArray to execute the query and start the download tasks.
Task<int>[] downloadTasks = downloadTasksQuery.ToArray();
5. Por fim, use o método Sum para obter a soma dos comprimentos de todos os sites. Adicione a seguinte
linha ao SumPageSizesAsync .
Exemplo
O código a seguir mostra as extensões para o projeto que usa o método GetURLContentsAsync para baixar
conteúdo da Web.
// Add the following using directives, and add a reference for System.Net.Http.
using System.Net.Http;
using System.IO;
using System.Net;
namespace AsyncExampleWPF_WhenAll
{
public partial class MainWindow : Window
{
public MainWindow()
{
InitializeComponent();
}
// Create a query.
IEnumerable<Task<int>> downloadTasksQuery =
from url in urlList select ProcessURLAsync(url);
// Use ToArray to execute the query and start the download tasks.
Task<int>[] downloadTasks = downloadTasksQuery.ToArray();
// You can do other work here before awaiting.
//var total = 0;
//foreach (var url in urlList)
//{
// byte[] urlContents = await GetURLContentsAsync(url);
// DisplayResults(url, urlContents);
// The actions from the foreach loop are moved to this async method.
private async Task<int> ProcessURLAsync(string url)
{
var byteArray = await GetURLContentsAsync(url);
DisplayResults(url, byteArray);
return byteArray.Length;
}
Exemplo
O código a seguir mostra as extensões para o projeto que usa o método HttpClient.GetByteArrayAsync para
baixar conteúdo da Web.
// Add the following using directives, and add a reference for System.Net.Http.
using System.Net.Http;
using System.IO;
using System.Net;
namespace AsyncExampleWPF_HttpClient_WhenAll
{
public partial class MainWindow : Window
{
public MainWindow()
{
InitializeComponent();
}
// Create a query.
IEnumerable<Task<int>> downloadTasksQuery =
from url in urlList select ProcessURLAsync(url, client);
// Use ToArray to execute the query and start the download tasks.
Task<int>[] downloadTasks = downloadTasksQuery.ToArray();
//var total = 0;
//foreach (var url in urlList)
//{
// // GetByteArrayAsync returns a Task<T>. At completion, the task
// // produces a byte array.
// byte[] urlContent = await client.GetByteArrayAsync(url);
// DisplayResults(url, urlContent);
// The actions from the foreach loop are moved to this async method.
async Task<int> ProcessURLAsync(string url, HttpClient client)
{
byte[] byteArray = await client.GetByteArrayAsync(url);
DisplayResults(url, byteArray);
return byteArray.Length;
}
}
Consulte também
Task.WhenAll
Passo a passo: acessando a Web e usando async e await (C#)
Como fazer várias solicitações da Web em paralelo
usando Async e Await (C#)
25/11/2019 • 9 minutes to read • Edit Online
Em um método assíncrono, as tarefas são iniciadas quando elas são criadas. O operador await é aplicado à tarefa
no ponto do método em que o processamento não pode continuar até que a tarefa seja concluída. Geralmente,
uma tarefa é aguardada assim que ela é criada, como mostrado no exemplo a seguir.
No entanto, você pode separar a criação da tarefa da espera da tarefa se o programa tiver outro trabalho a
realizar, que não depende da conclusão da tarefa.
// While the task is running, you can do other work that doesn't depend
// on the results of the task.
// . . . . .
// The application of await suspends the rest of this method until the task is complete.
var result = await myTask;
Entre o início de uma tarefa e a espera por ela, você pode iniciar outras tarefas. As tarefas adicionais são
executadas implicitamente em paralelo, mas não são criados threads adicionais.
O programa a seguir inicia três downloads assíncronos na Web e, em seguida, os aguarda na ordem em que
foram chamados. Observe ao executar o programa, que as tarefas nem sempre são concluídas na ordem em que
foram criadas e aguardadas. Eles começam a ser executadas quando são criadas e uma ou mais tarefas podem
terminar antes que o método alcance as expressões await.
NOTE
Para concluir esse projeto, você precisa ter o Visual Studio 2012 ou posterior e o .NET Framework 4.5 ou posterior
instalados no seu computador.
Para obter outro exemplo que inicia várias tarefas ao mesmo tempo, consulte como estender o passo a
assíncrona usando Task. WhenAll (C#).
Você pode baixar o código deste exemplo de Exemplos de código para desenvolvedores.
Para configurar o projeto
1. Para configurar um aplicativo WPF, complete as etapas a seguir. Você pode encontrar instruções
detalhadas dessas etapas em Passo a passo: acessando a Web usando async e await (C#).
Crie um aplicativo WPF que contenha uma caixa de texto e um botão. Dê o nome startButton para
o botão e resultsTextBox , para a caixa de texto.
Adicione uma referência para System.Net.Http.
No arquivo MainWindow.xaml.cs, adicione uma diretiva using para System.Net.Http .
Para adicionar o código
1. Na janela de design, MainWindow.xaml, clique duas vezes no botão para criar o manipulador de eventos
startButton_Click no MainWindow.xaml.cs.
resultsTextBox.Clear();
await CreateMultipleTasksAsync();
resultsTextBox.Text += "\r\n\r\nControl returned to startButton_Click.\r\n";
Exemplo
O código a seguir contem o exemplo completo.
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Data;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Imaging;
using System.Windows.Navigation;
using System.Windows.Shapes;
// Add the following using directive, and add a reference for System.Net.Http.
using System.Net.Http;
namespace AsyncExample_MultipleTasks
{
public partial class MainWindow : Window
{
public MainWindow()
{
InitializeComponent();
}
}
Consulte também
Passo a passo: acessando a Web e usando async e await (C#)
Programação assíncrona com async e await (C#)
Como estender a instrução assíncrona usando Task. WhenAll (C#)
Tipos de retorno assíncronos (C#)
23/10/2019 • 12 minutes to read • Edit Online
return leisureHours;
}
}
// The example displays output like the following:
// Today is Wednesday, May 24, 2017
// Today's hours of leisure: 5
// </Snippet >
IMPORTANT
A propriedade Result é uma propriedade de bloqueio. Se você tentar acessá-la antes que sua tarefa seja concluída, o
thread que está ativo no momento será bloqueado até que a tarefa seja concluída e o valor esteja disponível. Na maioria
dos casos, você deve acessar o valor usando await em vez de acessar a propriedade diretamente.
O exemplo anterior recuperou o valor da propriedade Result para bloquear o thread principal, de modo que o método
ShowTodaysInfo pudesse concluir a execução antes do encerramento do aplicativo.
var integerTask = GetLeisureHours();
// You can do other work that does not rely on integerTask before awaiting.
using System;
using System.Threading.Tasks;
O WaitAndApologize é aguardado, usando uma instrução await, em vez de uma expressão await, semelhante à
instrução de chamada a um método síncrono de retorno void. A aplicação de um operador await, nesse caso,
não produz um valor.
Como no exemplo Task<TResult> anterior, você pode separar a chamada ao WaitAndApologize , da aplicação de
um operador await, como mostrado no código a seguir. No entanto, lembre-se que uma Task não tem uma
propriedade Result e que nenhum valor será produzido quando um operador await for aplicado a uma Task .
O código a seguir separa a chamada ao método WaitAndApologize da espera pela tarefa que o método retorna.
using System;
using System.Threading.Tasks;
await secondHandlerFinished;
}
// Expected output:
// About to click a button...
// Somebody has clicked a button. Let's raise the event...
// Handler 1 is starting...
// Handler 1 is done.
// Handler 2 is starting...
// Handler 2 is about to go async...
// Handler 3 is starting...
// Handler 3 is done.
// All listeners are notified.
// Button's Click method returned.
// Handler 2 is done.
Já que Task e Task<TResult> são tipos de referência, a alocação de memória em caminhos críticos para o
desempenho, especialmente quando alocações ocorrerem em loops estreitos, podem afetar o desempenho.
Suporte para tipos de retorno generalizados significa que você pode retornar um tipo de valor leve em vez de
um tipo de referência para evitar as alocações de memória adicionais.
O .NET fornece a estrutura System.Threading.Tasks.ValueTask<TResult> como uma implementação leve de um
valor de retorno de tarefa generalizado. Para usar o tipo System.Threading.Tasks.ValueTask<TResult>, você
deve adicionar o pacote NuGet System.Threading.Tasks.Extensions ao seu projeto. O exemplo a seguir usa a
estrutura ValueTask<TResult> para recuperar o valor de dois lançamentos de dados.
using System;
using System.Threading.Tasks;
class Program
{
static Random rnd;
await Task.Delay(500);
int diceRoll = rnd.Next(1, 7);
return diceRoll;
}
}
// The example displays output like the following:
// ...Shaking the dice...
// You rolled 8
Consulte também
FromResult
Passo a passo: acesso à Web com o uso de Async e Await (C#)
Fluxo de controle em programas assíncronos (C#)
async
await
Fluxo de controle em programas assíncronos (C#)
23/10/2019 • 16 minutes to read • Edit Online
Você pode escrever e manter programas assíncronos mais facilmente usando as palavras-chave async e await .
No entanto, os resultados podem surpreendê-lo se você não entender o funcionamento do seu programa. Este
tópico rastreia o fluxo de controle por meio de um programa assíncrono simples para mostrar quando o
controle se move de um método para o outro e quais informações são transferidas a cada vez.
Em geral, você marca os métodos que contêm código assíncrono com o modificador async (C#). Em um método
marcado com um modificador assíncrono, você pode usar um operador await (C#) para especificar o local em
que o método faz uma pausa para esperar pela conclusão de um processo assíncrono que foi chamado. Para
obter mais informações, consulte Programação assíncrona com async e await (C#).
O exemplo a seguir usa os métodos assíncronos para baixar o conteúdo de um site especificado como uma
cadeia de caracteres e exibir o comprimento da cadeia de caracteres. O exemplo contém os dois métodos a
seguir.
startButton_Click , que chama AccessTheWebAsync e exibe o resultado.
AccessTheWebAsync , que baixa o conteúdo de um site na forma de uma cadeia de caracteres e retorna o
comprimento da cadeia de caracteres. AccessTheWebAsync usa um método HttpClient assíncrono,
GetStringAsync(String), para baixar o conteúdo.
Linhas numeradas de exibição aparecem em pontos estratégicos em todo o programa para ajudá-lo a entender
como o programa é executado e explicar o que acontece em cada ponto marcado. As linhas de exibição são
rotuladas como "UM"a "SEIS". Os rótulos representam a ordem na qual o programa alcança essas linhas de
código.
O código a seguir mostra uma estrutura de tópicos do programa.
public partial class MainWindow : Window
{
// . . .
private async void startButton_Click(object sender, RoutedEventArgs e)
{
// ONE
Task<int> getLengthTask = AccessTheWebAsync();
// FOUR
int contentLength = await getLengthTask;
// SIX
resultsTextBox.Text +=
$"\r\nLength of the downloaded string: {contentLength}.\r\n";
}
// THREE
string urlContents = await getStringTask;
// FIVE
return urlContents.Length;
}
}
Cada um dos locais rotulados, "UM"a "SEIS," exibe informações sobre o estado atual do programa. A saída a
seguir será produzida:
Configurar o programa
Você pode baixar o código usado nesse tópico no MSDN ou você mesmo pode criá-lo.
NOTE
Para executar o exemplo, você deve ter o Visual Studio 2012 ou mais recente e o .NET Framework 4.5 ou posterior
instalados no seu computador.
Baixar o programa
Você pode baixar o aplicativo para este tópico em Amostra assíncrona: Fluxo de controle em programas
assíncronos. As etapas a seguir abrem e executam o programa.
1. Descompacte o arquivo baixado e, em seguida, inicie o Visual Studio.
2. Na barra de menus, escolha Arquivo > Abrir > Projeto/Solução.
3. Navegue até a pasta que contém o código de exemplo descompactado, abra o arquivo da solução (.sln) e,
em seguida, escolha a tecla F5 para compilar e executar o projeto.
Crie o programa sozinho
O projeto WPF (Windows Presentation Foundation) a seguir contém o exemplo de código deste tópico.
Para executar o projeto, realize as seguintes etapas:
1. Inicie o Visual Studio.
2. Na barra de menus, selecione Arquivo > Novo > Projeto.
A caixa de diálogo Novo Projeto é aberta.
3. Escolha a categoria Instalado > Visual C# > Área de Trabalho do Windows e, em seguida, escolha
Aplicativo WPF na lista de modelos de projeto.
4. Digite AsyncTracer como o nome do projeto e, em seguida, escolha o botão OK.
O novo projeto aparece no Gerenciador de Soluções.
5. No Editor do Visual Studio Code, escolha a guia MainWindow.xaml.
Se a guia não estiver visível, abra o menu de atalho para MainWindow.xaml no Gerenciador de
Soluções e, em seguida, escolha Exibir Código.
6. Na exibição XAML de MainWindow.xaml, substitua o código pelo código a seguir.
<Window
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" mc:Ignorable="d"
x:Class="AsyncTracer.MainWindow"
Title="Control Flow Trace" Height="350" Width="592">
<Grid>
<Button x:Name="startButton" Content="Start
" HorizontalAlignment="Left" Margin="250,10,0,0" VerticalAlignment="Top" Width="75" Height="24"
Click="startButton_Click" d:LayoutOverrides="GridBox"/>
<TextBox x:Name="resultsTextBox" HorizontalAlignment="Left" TextWrapping="Wrap"
VerticalAlignment="Bottom" Width="576" Height="265" FontFamily="Lucida Console" FontSize="10"
VerticalScrollBarVisibility="Visible" Grid.ColumnSpan="3"/>
</Grid>
</Window>
Uma janela simples, contendo uma caixa de texto e um botão, aparecerá no modo de exibição de Design
de MainWindow.xaml.
7. Adicione uma referência para System.Net.Http.
8. No Gerenciador de Soluções, abra o menu de atalho de MainWindow.xaml.cs e, em seguida, escolha
Exibir Código.
9. Em MainWindow.xaml.cs, substitua o código pelo código a seguir.
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Data;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Imaging;
using System.Windows.Navigation;
using System.Windows.Shapes;
namespace AsyncTracer
{
public partial class MainWindow : Window
{
public MainWindow()
{
InitializeComponent();
}
resultsTextBox.Text +=
$"\r\nLength of the downloaded string: {contentLength}.\r\n";
}
resultsTextBox.Text +=
"\r\n About to await getStringTask and return a Task<int> to
startButton_Click.\r\n";
return urlContents.Length;
}
}
}
Rastrear o programa
Etapas UM e DOIS
As duas primeiras linhas de exibição rastreiam o caminho conforme startButton_Click chama
AccessTheWebAsync e AccessTheWebAsync chama o método HttpClient assíncrono GetStringAsync( String). A
imagem a seguir delineia as chamadas de método a método.
O tipo de retorno de ambos AccessTheWebAsync e client.GetStringAsync é Task<TResult>. Para
AccessTheWebAsync , TResult é um inteiro. Para GetStringAsync , TResult é uma cadeia de caracteres. Para obter
mais informações sobre tipos de retorno de método assíncrono, consulte Tipos de retorno assíncronos (C#).
Um método assíncrono de retorno de tarefas retorna uma instância de tarefa, quando o controle volta para o
chamador. O controle retorna de um método assíncrono para seu chamador quando um operador await é
encontrado no método chamado ou quando o método chamado termina. As linhas de exibição rotuladas como
"TRÊS"até "SEIS" rastreiam essa parte do processo.
Etapa TRÊS
Em AccessTheWebAsync , o método assíncrono GetStringAsync(String) é chamado para baixar o conteúdo de
página da Web de destino. O controle retorna de client.GetStringAsync para AccessTheWebAsync quando
client.GetStringAsync retornar.
O método client.GetStringAsync retorna uma tarefa de cadeia de caracteres que é atribuída à variável
getStringTask em AccessTheWebAsync . A linha do programa de exemplo a seguir mostra a chamada à
client.GetStringAsync e a atribuição.
Você pode imaginar a tarefa como uma promessa de client.GetStringAsync para eventualmente produzir uma
cadeia de caracteres real. Enquanto isso, se AccessTheWebAsync tiver trabalho a fazer, que não dependa da cadeia
de caracteres prometida de client.GetStringAsync , o trabalho poderá continuar enquanto
client.GetStringAsync aguarda. No exemplo, as linhas de saída seguintes, que são rotuladas como "TRÊS",
representam a oportunidade para fazer o trabalho independente
NOTE
Normalmente, você aguarda a chamada a um método assíncrono imediatamente. Por exemplo, a atribuição a seguir
poderia substituir o código anterior que cria e, em seguida, aguarda getStringTask :
string urlContents = await client.GetStringAsync("https://msdn.microsoft.com");
Neste tópico, o operador await é aplicado posteriormente para acomodar as linhas de saída que marcam o fluxo de
controle em todo o programa.
Etapa QUATRO
O tipo de retorno declarado de AccessTheWebAsync é Task<int> . Portanto, quando AccessTheWebAsync for
suspenso, ele retornará uma tarefa de inteiro para startButton_Click . Você deve compreender que a tarefa
retornada não é getStringTask . A tarefa retornada é uma nova tarefa de um inteiro que representa o que ainda
precisa ser feito no método suspenso AccessTheWebAsync . A tarefa é uma promessa de AccessTheWebAsync para
produzir um inteiro quando a tarefa for concluída.
A instrução a seguir atribui essa tarefa à variável getLengthTask .
Como em AccessTheWebAsync , startButton_Click pode continuar com o trabalho que não dependa dos
resultados da tarefa assíncrona ( getLengthTask ) até que a tarefa seja aguardada. As seguintes linhas de saída
representam esse trabalho.
O avanço em startButton_Click será suspenso quando getLengthTask for aguardada. A seguinte instrução de
atribuição suspende startButton_Click até que AccessTheWebAsync seja concluída.
Na ilustração a seguir, as setas mostram o fluxo de controle da expressão await em AccessTheWebAsync para a
atribuição de um valor para getLengthTask , seguido pelo processamento normal de startButton_Click até que
getLengthTask seja aguardada.
Etapa CINCO
Quando client.GetStringAsync sinaliza que está concluído, o processamento em AccessTheWebAsync é liberado
da suspensão e pode continuar após a instrução await. As seguintes linhas de saída representam a retomada do
processamento.
A expressão await recupera de getLengthTask o valor inteiro, que é o operando da instrução return em
AccessTheWebAsync . A instrução a seguir atribui esse valor à variável contentLength .
É possível adicionar flexibilidade e precisão a seus aplicativos assíncronos usando os métodos e propriedades
que o tipo Task disponibiliza. Os tópicos nesta seção mostram exemplos que usam CancellationToken e métodos
de Task importantes como Task.WhenAll e Task.WhenAny.
Usando WhenAny e WhenAll , é possível, com facilidade, iniciar várias tarefas e aguardar sua conclusão
monitorando uma única tarefa.
WhenAny retorna uma tarefa que é concluída quando qualquer tarefa em uma coleção for concluída.
Para obter exemplos que usam WhenAny , consulte Cancelar as demais tarefas assíncronas depois que uma
delas estiver concluída (C#) e Iniciar várias tarefas assíncronas e processá-las na conclusão (C#).
WhenAll retorna uma tarefa que é concluída quando todas as tarefas em uma coleção forem concluídas.
Para obter mais informações e um exemplo que usa WhenAll , consulte como estender a explicação
assíncrona usando Task. WhenAll (C#).
Esta seção inclui os seguintes exemplos.
Cancelar uma tarefa assíncrona ou uma lista de tarefas (C#).
Cancelar tarefas assíncronas após um período (C#)
Cancelar as demais tarefas assíncronas depois que uma delas estiver concluída (C#)
Iniciar várias tarefas assíncronas e processá-las na conclusão (C#)
NOTE
Para executar os exemplos, você precisa ter o Visual Studio 2012 ou uma versão mais recente e o .NET Framework 4.5 ou
posterior instalados em seu computador.
Os projetos criam uma interface do usuário que contém um botão que inicia o processo e um botão que o
cancela, como mostra a imagem a seguir. Os botões são chamados startButton e cancelButton .
É possível baixar projetos completos do WPF (Windows Presentation Foundation) em Exemplo assíncrono:
ajuste fino de seu aplicativo.
Consulte também
Programação assíncrona com async e await (C#)
Cancelar uma tarefa assíncrona ou uma lista de
tarefas (C#)
23/10/2019 • 14 minutes to read • Edit Online
Você pode configurar um botão que pode ser usado para cancelar um aplicativo assíncrono se não desejar
aguardar sua conclusão. Seguindo os exemplos neste tópico, você pode adicionar um botão de cancelamento a
um aplicativo que baixa o conteúdo de um site ou uma lista de sites.
Os exemplos usam a interface do usuário que Ajuste fino de seu aplicativo assíncrono (C#) descreve.
NOTE
Para executar os exemplos, você precisa ter o Visual Studio 2012 ou uma versão mais recente e o .NET Framework 4.5 ou
posterior instalados em seu computador.
TIP
Se você não quiser baixar o projeto, você poderá examinar o arquivo MainWindow.xaml.cs no final deste tópico.
Criar o exemplo
As alterações a seguir adicionam um botão Cancelar a um aplicativo que baixa um site. Se não desejar baixar ou
compilar o exemplo, você poderá examinar o produto final na seção "Exemplos completos" no final deste tópico.
Os asteriscos marcam as alterações no código.
Para compilar o exemplo você mesmo, passo a passo, siga as instruções na seção “Baixando o exemplo”, mas
escolha StarterCode como o Projeto de Inicialização em vez de CancelATask.
Em seguida, adicione as seguintes alterações ao arquivo MainWindow.xaml.cs desse projeto.
1. Declare uma variável CancellationTokenSource , cts , que está no escopo para todos os métodos a
acessarem.
2. Adicione o seguinte manipulador de eventos para o botão Cancelar. O manipulador de eventos usa o
método CancellationTokenSource.Cancel para notificar cts quando o usuário solicita o cancelamento.
try
{
// ***Send a token to carry the message if cancellation is requested.
int contentLength = await AccessTheWebAsync(cts.Token);
resultsTextBox.Text += $"\r\nLength of the downloaded string: {contentLength}.\r\n";
}
// *** If cancellation is requested, an OperationCanceledException results.
catch (OperationCanceledException)
{
resultsTextBox.Text += "\r\nDownload canceled.\r\n";
}
catch (Exception)
{
resultsTextBox.Text += "\r\nDownload failed.\r\n";
}
Ready to download.
Length of the downloaded string: 158125.
Se você escolher o botão Cancelar antes de o programa terminar de baixar o conteúdo, o programa
produzirá a saída a seguir.
Ready to download.
Download canceled.
3. Adicione o seguinte loop em AccessTheWebAsync para processar cada endereço web na lista.
resultsTextBox.Text +=
$"\r\nLength of the downloaded string: {urlContents.Length}.\r\n";
}
4. Como AccessTheWebAsync exibe os comprimentos, o método não precisa retornar nada. Remova a
instrução de retorno e altere o tipo de retorno do método para Task em vez de Task<TResult>.
await AccessTheWebAsync(cts.Token);
Downloads complete.
Se você escolher o botão Cancelar antes de os downloads serem concluídos, a saída conterá os tamanhos
dos downloads que foram concluídos antes do cancelamento.
Downloads canceled.
Exemplos completos
As seções a seguir contêm o código para cada um dos exemplos anteriores. Observe que você deve adicionar
uma referência para System.Net.Http.
Você pode baixar os projetos em Amostra assíncrona: Ajustando o aplicativo.
Exemplo – cancelar uma tarefa
O código a seguir é o arquivo MainWindow.xaml.cs completo do exemplo que cancela uma única tarefa.
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Data;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Imaging;
using System.Windows.Navigation;
using System.Windows.Shapes;
using System.Threading;
namespace CancelATask
{
public partial class MainWindow : Window
{
{
// ***Declare a System.Threading.CancellationTokenSource.
CancellationTokenSource cts;
public MainWindow()
{
InitializeComponent();
}
resultsTextBox.Clear();
try
{
// ***Send a token to carry the message if cancellation is requested.
int contentLength = await AccessTheWebAsync(cts.Token);
resultsTextBox.Text +=
$"\r\nLength of the downloaded string: {contentLength}.\r\n";
}
// *** If cancellation is requested, an OperationCanceledException results.
catch (OperationCanceledException)
{
resultsTextBox.Text += "\r\nDownload canceled.\r\n";
}
catch (Exception)
{
resultsTextBox.Text += "\r\nDownload failed.\r\n";
}
// Ready to download.
// Ready to download.
// Download canceled.
}
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Data;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Imaging;
using System.Windows.Navigation;
using System.Windows.Shapes;
namespace CancelAListOfTasks
{
public partial class MainWindow : Window
{
// Declare a System.Threading.CancellationTokenSource.
CancellationTokenSource cts;
public MainWindow()
{
InitializeComponent();
}
resultsTextBox.Clear();
try
{
await AccessTheWebAsync(cts.Token);
// ***Small change in the display lines.
resultsTextBox.Text += "\r\nDownloads complete.";
}
catch (OperationCanceledException)
{
resultsTextBox.Text += "\r\nDownloads canceled.";
}
catch (Exception)
{
resultsTextBox.Text += "\r\nDownloads failed.";
}
resultsTextBox.Text +=
$"\r\nLength of the downloaded string: {urlContents.Length}.\r\n";
}
}
//Downloads complete.
//Downloads canceled.
}
Consulte também
CancellationTokenSource
CancellationToken
Programação assíncrona com async e await (C#)
Ajuste fino de seu aplicativo assíncrono (C#)
Exemplo de Async: ajuste do seu aplicativo
Cancelar tarefas assíncronas após um período (C#)
23/10/2019 • 6 minutes to read • Edit Online
Você pode cancelar uma operação assíncrona após um período de tempo usando o método
CancellationTokenSource.CancelAfter se você não deseja aguardar a conclusão da operação. Esse método agenda
o cancelamento de quaisquer tarefas associadas que não são concluídas dentro do período designado pela
expressão CancelAfter .
Este exemplo adiciona o código desenvolvido em Cancelar uma tarefa assíncrona ou uma lista de tarefas (C#)
para baixar uma lista de sites e para exibir o tamanho dos conteúdos de cada um.
NOTE
Para executar os exemplos, você precisa ter o Visual Studio 2012 ou uma versão mais recente e o .NET Framework 4.5 ou
posterior instalados em seu computador.
Baixar o exemplo
Baixe o projeto completo do WPF (Windows Presentation Foundation) em Amostra assíncrona: Ajustando o
aplicativo e, em seguida, siga estas etapas.
1. Descompacte o arquivo baixado e, em seguida, inicie o Visual Studio.
2. Na barra de menus, escolha Arquivo > Abrir > Projeto/Solução.
3. Na caixa de diálogo Abrir Projeto, abra a pasta em que está o código de exemplo que você descompactou
e, em seguida, abra o arquivo de solução (.sln) de AsyncFineTuningCS.
4. No Gerenciador de Soluções, abra o menu de atalho do projeto CancelAfterTime e, em seguida,
escolha Definir como Projeto de Inicialização.
5. Pressione a tecla F5 para executar o projeto. (Ou pressione Ctrl+F5 para executar o projeto sem depurá-
lo).
6. Execute o programa várias vezes para verificar que a saída pode mostrar a saída para todos os sites,
nenhum site ou alguns sites.
Se você não quiser baixar o projeto, você poderá examinar o arquivo MainWindow.xaml.cs no final deste tópico.
Criar o exemplo
O exemplo neste tópico adiciona ao projeto que é desenvolvido em Cancelar uma tarefa assíncrona ou uma lista
de tarefas (C#) para cancelar uma lista de tarefas. O exemplo usa a mesma interface do usuário, embora o botão
Cancelar não seja explicitamente usado.
Para compilar o exemplo você mesmo, passo a passo, siga as instruções na seção "Baixando o exemplo", mas
escolha CancelAListOfTasks como o Projeto de Inicialização. Adicione as alterações deste tópico ao projeto.
Para especificar um tempo máximo antes que as tarefas sejam marcadas como canceladas, adicione uma chamada
para CancelAfter a startButton_Click , como o exemplo a seguir mostra. A adição é marcada com asteriscos.
private async void startButton_Click(object sender, RoutedEventArgs e)
{
// Instantiate the CancellationTokenSource.
cts = new CancellationTokenSource();
resultsTextBox.Clear();
try
{
// ***Set up the CancellationTokenSource to cancel after 2.5 seconds. (You
// can adjust the time.)
cts.CancelAfter(2500);
await AccessTheWebAsync(cts.Token);
resultsTextBox.Text += "\r\nDownloads succeeded.\r\n";
}
catch (OperationCanceledException)
{
resultsTextBox.Text += "\r\nDownloads canceled.\r\n";
}
catch (Exception)
{
resultsTextBox.Text += "\r\nDownloads failed.\r\n";
}
cts = null;
}
Execute o programa várias vezes para verificar que a saída pode mostrar a saída para todos os sites, nenhum site
ou alguns sites. A saída a seguir é um exemplo.
Downloads canceled.
Exemplo completo
O código a seguir é o texto completo do arquivo MainWindow.xaml.cs para o exemplo. Os asteriscos marcam os
elementos que foram adicionados para esse exemplo.
Observe que você deve adicionar uma referência para System.Net.Http.
Baixe o projeto em Amostra assíncrona: Ajustando o aplicativo.
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Data;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Imaging;
using System.Windows.Navigation;
using System.Windows.Shapes;
// Add a using directive and a reference for System.Net.Http.
using System.Net.Http;
namespace CancelAfterTime
{
public partial class MainWindow : Window
{
// Declare a System.Threading.CancellationTokenSource.
CancellationTokenSource cts;
public MainWindow()
{
InitializeComponent();
}
resultsTextBox.Clear();
try
{
// ***Set up the CancellationTokenSource to cancel after 2.5 seconds. (You
// can adjust the time.)
cts.CancelAfter(2500);
await AccessTheWebAsync(cts.Token);
resultsTextBox.Text += "\r\nDownloads succeeded.\r\n";
}
catch (OperationCanceledException)
{
resultsTextBox.Text += "\r\nDownloads canceled.\r\n";
}
catch (Exception)
{
resultsTextBox.Text += "\r\nDownloads failed.\r\n";
}
cts = null;
}
resultsTextBox.Text +=
$"\r\nLength of the downloaded string: {urlContents.Length}.\r\n";
}
}
// Sample Output:
// Downloads canceled.
}
Consulte também
Programação assíncrona com async e await (C#)
Passo a passo: acesso à Web com o uso de Async e Await (C#)
Cancelar uma tarefa assíncrona ou uma lista de tarefas (C#)
Ajuste fino de seu aplicativo assíncrono (C#)
Exemplo de Async: ajuste do seu aplicativo
Cancelar as demais tarefas assíncronas depois que
uma delas estiver concluída (C#)
23/10/2019 • 10 minutes to read • Edit Online
Usando o método Task.WhenAny juntamente com um CancellationToken, você pode cancelar todas as tarefas
restantes quando uma tarefa é concluída. O método WhenAny leva um argumento que é uma coleção de tarefas.
O método inicia todas as tarefas e retorna uma única tarefa. A tarefa única será concluída quando qualquer tarefa
na coleção for concluída.
Este exemplo demonstra como usar um token de cancelamento em conjunto com WhenAny para aguardar na
primeira tarefa que for finalizada na coleção de tarefas e para cancelar as tarefas restantes. Cada tarefa baixa o
conteúdo de um site. O exemplo exibe o comprimento do conteúdo do primeiro download que é concluído e
cancela os outros downloads.
NOTE
Para executar os exemplos, você precisa ter o Visual Studio 2012 ou uma versão mais recente e o .NET Framework 4.5 ou
posterior instalados em seu computador.
Baixando o Exemplo
Baixe o projeto completo do WPF (Windows Presentation Foundation) em Amostra assíncrona: Ajustando o
aplicativo e, em seguida, siga estas etapas.
1. Descompacte o arquivo baixado e, em seguida, inicie o Visual Studio.
2. Na barra de menus, escolha Arquivo, Abrir, Projeto/Solução.
3. Na caixa de diálogo Abrir Projeto, abra a pasta em que está o código de exemplo que você descompactou
e, em seguida, abra o arquivo de solução (.sln) de AsyncFineTuningCS.
4. No Gerenciador de Soluções, abra o menu de atalho do projeto CancelAfterOneTask e, em seguida,
escolha Definir como Projeto de Inicialização.
5. Escolha a tecla F5 para executar o projeto.
Escolha as teclas CTRL+F5 para executar o projeto sem depurá-lo.
6. Execute o programa várias vezes para verificar que diferentes downloads são concluídos em primeiro.
Se você não quiser baixar o projeto, você poderá examinar o arquivo MainWindow.xaml.cs no final deste tópico.
Compilando o Exemplo
O exemplo neste tópico adiciona ao projeto que é desenvolvido em Cancelar uma tarefa assíncrona ou uma lista
de tarefas (C#) para cancelar uma lista de tarefas. O exemplo usa a mesma interface do usuário, embora o botão
Cancelar não seja explicitamente usado.
Para compilar o exemplo você mesmo, passo a passo, siga as instruções na seção "Baixando o exemplo", mas
escolha CancelAListOfTasks como o Projeto de Inicialização. Adicione as alterações deste tópico ao projeto.
No arquivo MainWindow.xaml.cs do projeto CancelAListOfTasks, inicie a transição, movendo as etapas de
processamento de cada site do loop em AccessTheWebAsync para o seguinte método assíncrono.
// ***Bundle the processing steps for a website into one async method.
async Task<int> ProcessURLAsync(string url, HttpClient client, CancellationToken ct)
{
// GetAsync returns a Task<HttpResponseMessage>.
HttpResponseMessage response = await client.GetAsync(url, ct);
return urlContents.Length;
}
Em AccessTheWebAsync , este exemplo usa uma consulta, o método ToArray e o método WhenAny para criar e
iniciar uma matriz de tarefas. A aplicação de WhenAny à matriz retorna uma única tarefa que, quando colocada em
espera, resulta na primeira tarefa a alcançar a conclusão na matriz de tarefas.
Faça as seguintes alterações em AccessTheWebAsync . Os asteriscos marcam as alterações no arquivo de código.
1. Comente ou exclua o loop.
2. Crie uma consulta que, quando executada, produz uma coleção de tarefas genéricas. Cada chamada para
ProcessURLAsync retorna um Task<TResult> em que TResult é um inteiro.
3. Chame ToArray para executar a consulta e iniciar as tarefas. A aplicação do método WhenAny na próxima
etapa executaria a consulta e iniciaria as tarefas sem usar ToArray , mas outros métodos podem não fazê-
lo. A prática mais segura é forçar a execução da consulta explicitamente.
// ***Use ToArray to execute the query and start the download tasks.
Task<int>[] downloadTasks = downloadTasksQuery.ToArray();
// ***Call WhenAny and then await the result. The task that finishes
// first is assigned to firstFinishedTask.
Task<int> firstFinishedTask = await Task.WhenAny(downloadTasks);
5. Neste exemplo, você está interessado apenas na tarefa que termina primeiro. Portanto, use
CancellationTokenSource.Cancel para cancelar as tarefas restantes.
// ***Cancel the rest of the downloads. You just want the first one.
cts.Cancel();
Execute o programa várias vezes para verificar que diferentes downloads são concluídos em primeiro.
Exemplo completo
O código a seguir é o arquivo MainWindow.xaml.cs completo para o exemplo. Os asteriscos marcam os
elementos que foram adicionados para esse exemplo.
Observe que você deve adicionar uma referência para System.Net.Http.
Baixe o projeto em Amostra assíncrona: Ajustando o aplicativo.
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Data;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Imaging;
using System.Windows.Navigation;
using System.Windows.Shapes;
namespace CancelAfterOneTask
{
public partial class MainWindow : Window
{
// Declare a System.Threading.CancellationTokenSource.
CancellationTokenSource cts;
public MainWindow()
{
InitializeComponent();
}
resultsTextBox.Clear();
try
{
await AccessTheWebAsync(cts.Token);
resultsTextBox.Text += "\r\nDownload complete.";
}
catch (OperationCanceledException)
{
resultsTextBox.Text += "\r\nDownload canceled.";
}
catch (Exception)
{
{
resultsTextBox.Text += "\r\nDownload failed.";
}
// resultsTextBox.Text +=
// $"\r\nLength of the downloaded string: {urlContents.Length}.\r\n";
//}
// ***Use ToArray to execute the query and start the download tasks.
Task<int>[] downloadTasks = downloadTasksQuery.ToArray();
// ***Call WhenAny and then await the result. The task that finishes
// first is assigned to firstFinishedTask.
Task<int> firstFinishedTask = await Task.WhenAny(downloadTasks);
// ***Cancel the rest of the downloads. You just want the first one.
cts.Cancel();
// ***Bundle the processing steps for a website into one async method.
async Task<int> ProcessURLAsync(string url, HttpClient client, CancellationToken ct)
{
// GetAsync returns a Task<HttpResponseMessage>.
HttpResponseMessage response = await client.GetAsync(url, ct);
// Download complete.
}
Consulte também
WhenAny
Ajuste fino de seu aplicativo assíncrono (C#)
Programação assíncrona com async e await (C#)
Exemplo de Async: ajuste do seu aplicativo
Iniciar várias tarefas assíncronas e processá-las na
conclusão (C#)
23/10/2019 • 8 minutes to read • Edit Online
Usando Task.WhenAny, você pode iniciar várias tarefas ao mesmo tempo e processá-las individualmente
conforme elas foram concluídas, em vez de processá-las na ordem em que foram iniciadas.
O exemplo a seguir usa uma consulta para criar uma coleção de tarefas. Cada tarefa baixa o conteúdo de um site
especificado. Em cada iteração de um loop "while", uma chamada esperada para WhenAny retorna a tarefa na
coleção de tarefas que concluir o download primeiro. Essa tarefa é removida da coleção e processada. O loop é
repetido até que a coleção não contenha mais tarefas.
NOTE
Para executar os exemplos, você precisa ter o Visual Studio (2012 ou mais recente) e o .NET Framework 4.5 ou posterior
instalados no seu computador.
TIP
Se você não quiser baixar o projeto, será possível examinar o arquivo MainWindow.XAML.cs no final deste tópico em vez
disso.
1. Extraia os arquivos que você baixou do arquivo . zip e inicie o Visual Studio.
2. Na barra de menus, escolha Arquivo > Abrir > Projeto/Solução.
3. Na caixa de diálogo Abrir projeto , abra a pasta que contém o código de exemplo que você baixou e, em
seguida, abra o arquivo da solução ( . sln) para AsyncFineTuningCS/AsyncFineTuningVB.
4. No Gerenciador de Soluções, abra o menu de atalho do projeto ProcessTasksAsTheyFinish e escolha
Definir como Projeto de Inicialização.
5. Escolha a tecla F5 para executar o programa com depuração ou pressione Ctrl+F5 para executar o
programa sem depurá-lo.
6. Execute o projeto várias vezes para verificar se os tamanhos baixados não aparecem sempre na mesma
ordem.
Adicione um loop while que executa as seguintes etapas para cada tarefa na coleção:
1. Espera uma chamada para WhenAny para identificar a primeira tarefa na coleção a concluir o
download.
downloadTasks.Remove(firstFinishedTask);
3. Espera firstFinishedTask , que é retornado por uma chamada para ProcessURLAsync . A variável
firstFinishedTask é uma Task<TResult> em que TReturn é um inteiro. A tarefa já foi concluída,
mas você espera para recuperar o tamanho do site baixado, como mostra o exemplo a seguir. Se a
tarefa tiver falhado, await lançará a primeira exceção filha armazenada no AggregateException , ao
contrário da leitura da propriedade Result , que geraria o AggregateException .
Execute o programa várias vezes para verificar se os tamanhos baixados não aparecem sempre na mesma ordem.
Cau t i on
Você pode usar WhenAny em um loop, conforme descrito no exemplo, para resolver problemas que envolvem um
número pequeno de tarefas. No entanto, outras abordagens são mais eficientes se você tiver um número grande
de tarefas para processar. Para obter mais informações e exemplos, consulte Processando tarefas quando elas são
concluídas.
Exemplo completo
O código a seguir é o texto completo do arquivo MainWindow.XAML.cs para o exemplo. Os asteriscos marcam os
elementos que foram adicionados para esse exemplo. Além disso, observe que você deve adicionar uma
referência para System.Net.Http.
Baixe o projeto em Amostra assíncrona: Ajustando o aplicativo.
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Data;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Imaging;
using System.Windows.Navigation;
using System.Windows.Shapes;
namespace ProcessTasksAsTheyFinish
{
public partial class MainWindow : Window
{
// Declare a System.Threading.CancellationTokenSource.
CancellationTokenSource cts;
public MainWindow()
{
InitializeComponent();
}
try
{
await AccessTheWebAsync(cts.Token);
resultsTextBox.Text += "\r\nDownloads complete.";
}
catch (OperationCanceledException)
{
resultsTextBox.Text += "\r\nDownloads canceled.\r\n";
}
catch (Exception)
{
resultsTextBox.Text += "\r\nDownloads failed.\r\n";
}
cts = null;
}
// ***Add a loop to process the tasks one at a time until none remain.
while (downloadTasks.Count > 0)
{
// Identify the first task that completes.
Task<int> firstFinishedTask = await Task.WhenAny(downloadTasks);
// ***Remove the selected task from the list so that you don't
// process it more than once.
downloadTasks.Remove(firstFinishedTask);
return urlContents.Length;
}
}
}
// Sample Output:
Consulte também
WhenAny
Ajuste fino de seu aplicativo assíncrono (C#)
Programação assíncrona com async e await (C#)
Exemplo de Async: ajuste do seu aplicativo
Tratando a reentrada em aplicativos assíncronos (C#)
24/10/2019 • 28 minutes to read • Edit Online
Ao incluir código assíncrono em seu aplicativo, você deve considerar e, possivelmente, evitar a reentrância, que se
refere à reinserção de uma operação assíncrona antes de ela ser concluída. Se você não identificar e tratar as
possibilidades de reentrância, isso poderá causar resultados inesperados.
Neste tópico
Reconhecendo a reentrância
Tratando a reentrância
Desabilitar o botão Iniciar
Cancelar e reiniciar a operação
Executar várias operações e colocar a saída na fila
Examinar e executar o aplicativo de exemplo
NOTE
Para executar o exemplo, você deve ter o Visual Studio 2012 ou mais recente e o .NET Framework 4.5 ou posterior
instalados no seu computador.
NOTE
O protocolo TLS versão 1,2 agora é a versão mínima a ser usada no desenvolvimento de seu aplicativo. Se seu aplicativo for
destinado a uma versão do .NET Framework anterior a 4,7, consulte o artigo a seguir para práticas recomendadas de TLS
(Transport Layer Security) com o .NET Framework
Reconhecendo a reentrância
No exemplo deste tópico, os usuários escolhem um botão Iniciar para iniciar um aplicativo assíncrono que baixa
uma série de sites e calcula o número total de bytes baixados. Uma versão síncrona do exemplo responderia da
mesma forma, independentemente de quantas vezes um usuário escolhesse o botão porque, após a primeira vez,
o thread da interface do usuário ignora esses eventos até que o aplicativo conclua a execução. Em um aplicativo
assíncrono, no entanto, o thread da interface do usuário continua a responder e você pode reinserir a operação
assíncrona antes que ele seja concluído.
O exemplo a seguir mostra a saída esperada, caso o usuário escolha o botão Iniciar apenas uma vez. É exibida
uma lista dos sites baixados com o tamanho, em bytes, de cada site. O número total de bytes é exibido no final.
1. msdn.microsoft.com/library/hh191443.aspx 83732
2. msdn.microsoft.com/library/aa578028.aspx 205273
3. msdn.microsoft.com/library/jj155761.aspx 29019
4. msdn.microsoft.com/library/hh290140.aspx 117152
5. msdn.microsoft.com/library/hh524395.aspx 68959
6. msdn.microsoft.com/library/ms404677.aspx 197325
7. msdn.microsoft.com 42972
8. msdn.microsoft.com/library/ff730837.aspx 146159
No entanto, se o usuário escolhe o botão mais de uma vez, o manipulador de eventos é invocado repetidamente e
o processo de download é reinserido a cada vez. Como resultado, várias operações assíncronas estarão em
execução ao mesmo tempo, a saída intercalará os resultados e o número total de bytes será confuso.
1. msdn.microsoft.com/library/hh191443.aspx 83732
2. msdn.microsoft.com/library/aa578028.aspx 205273
3. msdn.microsoft.com/library/jj155761.aspx 29019
4. msdn.microsoft.com/library/hh290140.aspx 117152
5. msdn.microsoft.com/library/hh524395.aspx 68959
1. msdn.microsoft.com/library/hh191443.aspx 83732
2. msdn.microsoft.com/library/aa578028.aspx 205273
6. msdn.microsoft.com/library/ms404677.aspx 197325
3. msdn.microsoft.com/library/jj155761.aspx 29019
7. msdn.microsoft.com 42972
4. msdn.microsoft.com/library/hh290140.aspx 117152
8. msdn.microsoft.com/library/ff730837.aspx 146159
5. msdn.microsoft.com/library/hh524395.aspx 68959
1. msdn.microsoft.com/library/hh191443.aspx 83732
2. msdn.microsoft.com/library/aa578028.aspx 205273
6. msdn.microsoft.com/library/ms404677.aspx 197325
3. msdn.microsoft.com/library/jj155761.aspx 29019
4. msdn.microsoft.com/library/hh290140.aspx 117152
7. msdn.microsoft.com 42972
5. msdn.microsoft.com/library/hh524395.aspx 68959
8. msdn.microsoft.com/library/ff730837.aspx 146159
6. msdn.microsoft.com/library/ms404677.aspx 197325
7. msdn.microsoft.com 42972
8. msdn.microsoft.com/library/ff730837.aspx 146159
Você pode examinar o código que produz esta saída fazendo a rolagem até o final deste tópico. Você pode fazer
experimentos com o código baixando a solução em seu computador local e, em seguida, executar o projeto
WebsiteDownload ou usar o código no final deste tópico para criar seu próprio projeto. Para obter mais
informações e instruções, consulte Examinar e executar o aplicativo de exemplo.
Tratando a reentrância
É possível tratar a reentrância de várias maneiras, dependendo do que você deseja que seu aplicativo faça. Este
tópico apresenta os exemplos a seguir:
Desabilitar o botão Iniciar
Desabilitar o botão Iniciar enquanto a operação estiver em execução para que o usuário não possa
interrompê-la.
Cancelar e reiniciar a operação
Cancelar qualquer operação que ainda estiver em execução quando o usuário escolher o botão Iniciar
novamente e, em seguida, permitir que a operação solicitada mais recentemente continue.
Executar várias operações e colocar a saída na fila
Permitir que todas as operações solicitadas sejam executadas de forma assíncrona, mas coordenar a
exibição da saída, de forma que os resultados de cada operação apareçam juntos e em ordem.
Desabilitar o botão Iniciar
Você pode bloquear o botão Iniciar enquanto uma operação estiver em execução, desabilitando o botão na parte
superior do manipulador de eventos StartButton_Click . Você pode reativar o botão de dentro um bloco finally
quando a operação for concluída para que os usuários possam executar o aplicativo novamente.
Para configurar esse cenário, faça as seguintes alterações no código básico que é fornecido em Examinar e
executar o aplicativo de exemplo. Você também pode baixar o aplicativo finalizado de Exemplos assíncronos:
reentrância em aplicativos de área de trabalho do .NET. O nome do projeto é DisableStartButton.
try
{
await AccessTheWebAsync();
}
catch (Exception)
{
ResultsTextBox.Text += "\r\nDownloads failed.";
}
// ***Enable the Start button in case you want to run the program again.
finally
{
StartButton.IsEnabled = true;
}
}
Como resultado das alterações, o botão não responderá enquanto AccessTheWebAsync estiver baixando os sites,
para que o processo não possa ser reinserido.
Cancelar e reiniciar a operação
Em vez de desabilitar o botão Iniciar, você pode manter o botão ativo, mas, se o usuário escolher esse botão
novamente, cancele a operação que já está em execução e permita que a operação iniciada mais recentemente
continue.
Para obter mais informações sobre o cancelamento, consulte Ajuste fino de seu aplicativo assíncrono (C#).
Para configurar esse cenário, faça as seguintes alterações no código básico que é fornecido em Examinar e
executar o aplicativo de exemplo. Você também pode baixar o aplicativo finalizado de Exemplos assíncronos:
reentrância em aplicativos de área de trabalho do .NET. O nome do projeto é CancelAndRestart.
1. Declare uma variável CancellationTokenSource, cts , que está no escopo para todos os métodos.
public partial class MainWindow : Window // Or class MainPage
{
// *** Declare a System.Threading.CancellationTokenSource.
CancellationTokenSource cts;
2. Em StartButton_Click , determine se uma operação já está em andamento. Se o valor de cts for nulo,
nenhuma operação estará ativa. Se o valor não for nulo, a operação que já está em execução será
cancelada.
// *** Now set cts to a new value that you can use to cancel the current process
// if the button is chosen again.
CancellationTokenSource newCTS = new CancellationTokenSource();
cts = newCTS;
4. Ao final de StartButton_Click , o processo atual estará concluído, então, defina o valor de cts novamente
como nulo.
// *** When the process is complete, signal that another process can begin.
if (cts == newCTS)
cts = null;
O código a seguir mostra todas as alterações em StartButton_Click . As adições estão marcadas com asteriscos.
private async void StartButton_Click(object sender, RoutedEventArgs e)
{
// This line is commented out to make the results clearer in the output.
//ResultsTextBox.Clear();
// *** Now set cts to cancel the current process if the button is chosen again.
CancellationTokenSource newCTS = new CancellationTokenSource();
cts = newCTS;
try
{
// ***Send cts.Token to carry the message if there is a cancellation request.
await AccessTheWebAsync(cts.Token);
}
// *** Catch cancellations separately.
catch (OperationCanceledException)
{
ResultsTextBox.Text += "\r\nDownloads canceled.\r\n";
}
catch (Exception)
{
ResultsTextBox.Text += "\r\nDownloads failed.\r\n";
}
// *** When the process is complete, signal that another process can proceed.
if (cts == newCTS)
cts = null;
}
var total = 0;
var position = 0;
Se você escolher o botão Iniciar várias vezes enquanto este aplicativo estiver em execução, ele deverá produzir
resultados semelhantes à saída a seguir.
1. msdn.microsoft.com/library/hh191443.aspx 83732
2. msdn.microsoft.com/library/aa578028.aspx 205273
3. msdn.microsoft.com/library/jj155761.aspx 29019
4. msdn.microsoft.com/library/hh290140.aspx 122505
5. msdn.microsoft.com/library/hh524395.aspx 68959
6. msdn.microsoft.com/library/ms404677.aspx 197325
Download canceled.
1. msdn.microsoft.com/library/hh191443.aspx 83732
2. msdn.microsoft.com/library/aa578028.aspx 205273
3. msdn.microsoft.com/library/jj155761.aspx 29019
Download canceled.
1. msdn.microsoft.com/library/hh191443.aspx 83732
2. msdn.microsoft.com/library/aa578028.aspx 205273
3. msdn.microsoft.com/library/jj155761.aspx 29019
4. msdn.microsoft.com/library/hh290140.aspx 117152
5. msdn.microsoft.com/library/hh524395.aspx 68959
6. msdn.microsoft.com/library/ms404677.aspx 197325
7. msdn.microsoft.com 42972
8. msdn.microsoft.com/library/ff730837.aspx 146159
Para eliminar as listas parciais, remova a marca de comentário da primeira linha de código em StartButton_Click ,
para limpar a caixa de texto sempre que o usuário reiniciar a operação.
Executar várias operações e colocar a saída na fila
Este terceiro exemplo é mais complicado pois o aplicativo iniciará outra operação assíncrona a cada vez que o
usuário escolher o botão Iniciar e todas as operações serão executadas até a conclusão. Todas as operações
solicitadas baixam sites da lista de maneira assíncrona, mas a saída das operações será apresentada
sequencialmente. Ou seja, a atividade de download real é intercalada, conforme mostrado na saída em
Reconhecendo a reentrância, mas a lista de resultados para cada grupo é apresentada separadamente.
As operações compartilham uma Task global, pendingWork , que serve como um gatekeeper para o processo de
exibição.
Para configurar esse cenário, faça as seguintes alterações no código básico que é fornecido em Examinar e
executar o aplicativo de exemplo. Você também pode baixar o aplicativo finalizado de Exemplos assíncronos:
reentrância em aplicativos de área de trabalho do .NET. O nome do projeto é QueueResults.
A saída a seguir mostra o resultado, caso o usuário escolha o botão Iniciar apenas uma vez. O rótulo de letra A
indica que o resultado é da primeira vez que o botão Iniciar foi escolhido. Os números mostram a ordem das
URLs na lista de destinos de download.
#Starting group A.
#Task assigned for group A.
#Group A is complete.
Se o usuário escolher o botão Iniciar três vezes, o aplicativo produzirá uma saída semelhante à das linhas a
seguir. As linhas de informações que começam com um sinal de jogo da velha (#) rastreiam o progresso do
aplicativo.
#Starting group A.
#Task assigned for group A.
#Starting group B.
#Task assigned for group B.
#Starting group C.
#Task assigned for group C.
#Group A is complete.
#Group B is complete.
#Group C is complete.
Os grupos B e C iniciam antes da conclusão do grupo A, mas a saída para cada grupo é exibida separadamente.
Toda a saída para o grupo A aparece em primeiro, seguida por toda a saída para o grupo B e, em seguida, toda a
saída para o grupo C. O aplicativo sempre exibe os grupos em ordem e, para cada grupo, sempre exibe as
informações sobre os sites individuais na ordem em que as URLs aparecem na lista de URLs.
No entanto, não é possível prever a ordem na qual os downloads realmente acontecem. Depois que vários grupos
tiverem sido iniciados, as tarefas de download que eles geram estarão todos ativas. Você não pode presumir que
A-1 será baixado antes de B -1 e não pode presumir que A-1 será baixado antes de A-2.
Definições Globais
O código de exemplo contém as duas declarações globais a seguir, que estão visíveis em todos os métodos.
public partial class MainWindow : Window // Class MainPage in Windows Store app.
{
// ***Declare the following variables where all methods can access them.
private Task pendingWork = null;
private char group = (char)('A' - 1);
A variável Task , pendingWork , supervisiona o processo de exibição e impede que qualquer grupo interrompa a
operação de exibição do outro grupo. A variável de caractere, group , rotula a saída de grupos diferentes para
verificar se os resultados aparecem na ordem esperada.
O manipulador de eventos Click
O manipulador de eventos StartButton_Click , incrementa a letra de grupo sempre que o usuário escolhe o botão
Iniciar. Em seguida, o manipulador chama AccessTheWebAsync para executar a operação de download.
try
{
// *** Pass the group value to AccessTheWebAsync.
char finishedGroup = await AccessTheWebAsync(group);
// The following line verifies a successful return from the download and
// display procedures.
ResultsTextBox.Text += $"\r\n\r\n#Group {finishedGroup} is complete.\r\n";
}
catch (Exception)
{
ResultsTextBox.Text += "\r\nDownloads failed.";
}
}
O método AccessTheWebAsync
Este exemplo divide o AccessTheWebAsync em dois métodos. O primeiro método, AccessTheWebAsync , inicia todas
as tarefas de download de um grupo e configura pendingWork para controlar o processo de exibição. O método
usa uma LINQ (consulta integrada à linguagem) e um ToArray para iniciar todas as tarefas de download ao
mesmo tempo.
Em seguida, o AccessTheWebAsync chama FinishOneGroupAsync para aguardar a conclusão de cada download e
exibir seu comprimento.
O FinishOneGroupAsync retorna uma tarefa que é atribuída a pendingWork em AccessTheWebAsync . Esse valor
impede a interrupção por outra operação antes que a tarefa seja concluída.
private async Task<char> AccessTheWebAsync(char grp)
{
HttpClient client = new HttpClient();
// ***Kick off the downloads. The application of ToArray activates all the download tasks.
Task<byte[]>[] getContentTasks = urlList.Select(url => client.GetByteArrayAsync(url)).ToArray();
// ***Call the method that awaits the downloads and displays the results.
// Assign the Task that FinishOneGroupAsync returns to the gatekeeper task, pendingWork.
pendingWork = FinishOneGroupAsync(urlList, getContentTasks, grp);
ResultsTextBox.Text += $"\r\n#Task assigned for group {grp}. Download tasks are active.\r\n";
// ***This task is complete when a group has finished downloading and displaying.
await pendingWork;
O método FinishOneGroupAsync
Esse método percorre as tarefas de download em um grupo, aguardando cada uma delas, exibindo o
comprimento do site baixado e adicionando o comprimento ao total.
A primeira instrução em FinishOneGroupAsync usa pendingWork para verificar se o método de inserção não
interfere com uma operação que já está no processo de exibição ou que já está aguardando. Se houver uma
operação dessas em andamento, a operação de inserção deverá esperar por sua vez.
int total = 0;
Pontos de Interesse
As linhas de informações que começam com um sinal de jogo da velha (#) na saída esclarecem o funcionamento
deste exemplo.
A saída mostra os padrões a seguir.
Um grupo pode ser iniciado enquanto um grupo anterior estiver exibindo a saída, mas a exibição da saída
do grupo anterior não será interrompida.
#Starting group A.
#Task assigned for group A. Download tasks are active.
#Starting group B.
#Task assigned for group B. Download tasks are active.
#Group A is complete.
O tarefa pendingWork é nula no início de FinishOneGroupAsync somente para o grupo A, que foi iniciado
primeiro. O grupo A ainda não concluiu uma expressão await quando alcança o FinishOneGroupAsync .
Portanto, o controle não foi retornado ao AccessTheWebAsync e a primeira atribuição para pendingWork não
ocorreu.
As duas linhas a seguir sempre aparecem juntas na saída. O código nunca é interrompido entre o início de
uma operação de um grupo em StartButton_Click e a atribuição de uma tarefa para o grupo para
pendingWork .
#Starting group B.
#Task assigned for group B. Download tasks are active.
Depois que um grupo insere o StartButton_Click , a operação não conclui uma expressão await até que a
operação insira FinishOneGroupAsync . Portanto, nenhuma outra operação pode obter controle durante esse
segmento de código.
NOTE
Para executar o exemplo como um aplicativo da área de trabalho do WPF (Windows Presentation Foundation), você deve ter
o Visual Studio 2012 ou mais recente e o .NET Framework 4.5 ou posterior instalados no seu computador.
Baixar o aplicativo
1. Baixe o aplicativo compactado em Exemplos assíncronos: reentrância em aplicativos de área de trabalho do
.NET.
2. Descompacte o arquivo baixado e, em seguida, inicie o Visual Studio.
3. Na barra de menus, escolha Arquivo, Abrir, Projeto/Solução.
4. Navegue até a pasta que contém o código de exemplo descompactado e, em seguida, abra o arquivo de
solução (.sln).
5. No Gerenciador de Soluções, abra o menu de atalho do projeto que você deseja executar e, em seguida,
escolha Definir como Projeto de Inicialização.
6. Escolha as teclas CTRL+F5 para compilar e executar o projeto.
Compilando o aplicativo
A seção a seguir fornece o código para compilar o exemplo como um aplicativo do WPF.
Para c om pilar u m aplic at ivo do W P F
<Window x:Class="WebsiteDownloadWPF.MainWindow"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:local="using:WebsiteDownloadWPF"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
mc:Ignorable="d">
Uma janela simples, contendo uma caixa de texto e um botão, aparecerá no modo de exibição de Design
de MainWindow.xaml.
8. Em Gerenciador de soluções, clique com o botão direito do mouse em referências e selecione
Adicionar referência.
Adicione uma referência para System.Net.Http, se ela ainda não estiver selecionada.
9. No Gerenciador de Soluções, abra o menu de atalho de MainWindow.xaml.cs e, em seguida, escolha
Exibir Código.
10. Em MainWindow.xaml.cs, substitua o código pelo código a seguir.
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Data;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Imaging;
using System.Windows.Navigation;
using System.Windows.Shapes;
// Add the following using directives, and add a reference for System.Net.Http.
using System.Net.Http;
using System.Threading;
namespace WebsiteDownloadWPF
{
public partial class MainWindow : Window
{
public MainWindow()
{
System.Net.ServicePointManager.SecurityProtocol |= System.Net.SecurityProtocolType.Tls12;
InitializeComponent();
}
try
{
await AccessTheWebAsync();
}
catch (Exception)
{
ResultsTextBox.Text += "\r\nDownloads failed.";
}
}
var total = 0;
var position = 0;
11. Escolha a tecla CTRL+F5 para executar o programa e, em seguida, escolha o botão Iniciar várias vezes.
12. Faça as alterações de Desabilitar o botão Iniciar, Cancelar e reiniciar a operação ou Executar várias
operações e colocar a saída em fila para tratar a reentrância.
Consulte também
Passo a passo: acessando a Web e usando async e await (C#)
Programação assíncrona com async e await (C#)
Usando o Async para acessar arquivos (C#)
23/10/2019 • 9 minutes to read • Edit Online
Você pode usar o recurso async para acessar arquivos. Usando o recurso async, você pode chamar os métodos
assíncronos sem usar retornos de chamada ou dividir seu código em vários métodos ou expressões lambda. Para
tornar síncrono um código assíncrono, basta chamar um método assíncrono em vez de um método síncrono e
adicionar algumas palavras-chave ao código.
Você pode considerar seguintes motivos para adicionar a assincronia às chamadas de acesso ao arquivo:
A assincronia torna os aplicativos de interface do usuário mais responsivos porque o thread de interface do
usuário que inicia a operação pode executar outro trabalho. Se o thread de interface do usuário precisar
executar o código que leva muito tempo (por exemplo, mais de 50 milissegundos), a interface do usuário
poderá congelar até que a E/S seja concluída e o thread da interface do usuário possa processar entradas do
mouse e do teclado e outros eventos.
A assincronia melhora a escalabilidade do ASP.NET e outros aplicativos baseados em servidor reduzindo a
necessidade de threads. Se o aplicativo usar um thread dedicado por resposta e mil solicitações forem
tratadas simultaneamente, serão necessários mil threads. As operações assíncronas normalmente não
precisam usar um thread durante a espera. Elas podem usar o thread de conclusão de E/S existente
rapidamente no final.
A latência de uma operação de acesso de arquivo pode ser muito baixa nas condições atuais, mas a latência
pode aumentar consideravelmente no futuro. Por exemplo, um arquivo pode ser movido para um servidor
que está do outro lado do mundo.
A sobrecarga adicional de usar o recurso async é pequena.
As tarefas assíncronas podem facilmente ser executadas em paralelo.
Executando os exemplos
Para executar os exemplos neste tópico, você pode criar um Aplicativo WPF ou um Aplicativo do Windows
Forms e, em seguida, adicionar um Botão. No evento Click do botão, adicione uma chamada para o primeiro
método em cada exemplo.
Nos exemplos a seguir, inclua as seguintes instruções using .
using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.IO;
using System.Text;
using System.Threading.Tasks;
Gravando texto
O exemplo a seguir grava um texto em um arquivo. A cada instrução await, o método sai imediatamente. Quando
o arquivo de E/S for concluído, o método continuará na instrução após a instrução await. Observe que o
modificador async é a definição de métodos que usam a instrução await.
A primeira instrução retorna uma tarefa e faz com que o processamento do arquivo seja iniciado. A segunda
instrução com o await faz com que o método saia imediatamente e retorne uma tarefa diferente. Quando o
processamento do arquivo é concluído posteriormente, a execução retorna para a instrução após a await. Para
obter mais informações, consulte Fluxo de controle em programas assíncronos (C#).
Lendo texto
O exemplo a seguir lê o texto de um arquivo. O texto é armazenado em buffer e, nesse caso, colocado em um
StringBuilder. Diferentemente do exemplo anterior, a avaliação de await produz um valor.O método ReadAsync
retorna um Task<Int32>, portanto, a avaliação do await produz um valor Int32 ( numRead ) após a conclusão da
operação. Para obter mais informações, consulte Tipos de retorno assíncronos (C#).
public async Task ProcessReadAsync()
{
string filePath = @"temp2.txt";
if (File.Exists(filePath) == false)
{
Debug.WriteLine("file not found: " + filePath);
}
else
{
try
{
string text = await ReadTextAsync(filePath);
Debug.WriteLine(text);
}
catch (Exception ex)
{
Debug.WriteLine(ex.Message);
}
}
}
return sb.ToString();
}
}
try
{
for (int index = 1; index <= 10; index++)
{
string text = "In file " + index.ToString() + "\r\n";
tasks.Add(theTask);
}
await Task.WhenAll(tasks);
}
finally
{
foreach (FileStream sourceStream in sourceStreams)
{
sourceStream.Close();
}
}
}
Ao usar os métodos WriteAsync e ReadAsync, você pode especificar um CancellationToken, que pode ser usado
para cancelar o fluxo intermediário da operação. Para obter mais informações, consulte Ajuste fino de seu
aplicativo assíncrono (C#) e Cancelamento em threads gerenciados.
Consulte também
Programação assíncrona com async e await (C#)
Tipos de retorno assíncronos (C#)
Fluxo de controle em programas assíncronos (C#)
Atributos (C#)
25/11/2019 • 9 minutes to read • Edit Online
Usando atributos
Os atributos podem ser colocados em quase qualquer declaração, embora um atributo específico possa
restringir os tipos de declarações nas quais ele é válido. No C#, você especifica um atributo colocando o nome
do atributo entre colchetes ([]) acima da declaração da entidade à qual ele se aplica.
Neste exemplo, o atributo SerializableAttribute é usado para aplicar uma característica específica a uma classe:
[Serializable]
public class SampleClass
{
// Objects of this type can be serialized.
}
[System.Runtime.InteropServices.DllImport("user32.dll")]
extern static void SampleMethod();
Mais de um atributo pode ser colocado em uma declaração como o seguinte exemplo mostra:
using System.Runtime.InteropServices;
Alguns atributos podem ser especificados mais de uma vez para uma determinada entidade. Um exemplo de um
atributo multiuso é ConditionalAttribute:
[Conditional("DEBUG"), Conditional("TEST1")]
void TraceMethod()
{
// ...
}
NOTE
Por convenção, todos os nomes de atributo terminam com a palavra "Atributo" para distingui-los de outros itens nas
bibliotecas do .NET. No entanto, você não precisa especificar o sufixo de atributo ao usar atributos no código. Por exemplo,
[DllImport] é equivalente a [DllImportAttribute] , mas DllImportAttribute é o nome real do atributo na
Biblioteca de Classes .NET Framework.
Parâmetros de atributo
Muitos atributos têm parâmetros, que podem ser nomeados, sem nome ou posicionais. Quaisquer parâmetros
de posição devem ser especificados em uma determinada ordem e não podem ser omitidos. Parâmetros
nomeados são opcionais e podem ser especificados em qualquer ordem. Os parâmetros posicionais são
especificados primeiro. Por exemplo, esses três atributos são equivalentes:
[DllImport("user32.dll")]
[DllImport("user32.dll", SetLastError=false, ExactSpelling=false)]
[DllImport("user32.dll", ExactSpelling=false, SetLastError=false)]
O primeiro parâmetro, o nome da DLL, é posicional e sempre vir em primeiro lugar; os outros são nomeados.
Nesse caso, ambos os parâmetros nomeados são padronizados como false e, portanto, podem ser omitidos.
Parâmetros de posição correspondem aos parâmetros do construtor de atributo. Parâmetros nomeados ou
opcionais correspondem a propriedades ou a campos do atributo. Consulte a documentação do atributo
individual para obter informações sobre valores de parâmetro padrão.
Destinos do atributo
O destino de um atributo é a entidade à qual o atributo se aplica. Por exemplo, um atributo pode ser aplicado a
uma classe, um método específico ou um assembly inteiro. Por padrão, um atributo se aplica ao elemento que o
segue. Mas você pode identificar explicitamente, por exemplo, se um atributo é aplicado a um método, ou a seu
parâmetro ou a seu valor retornado.
Para identificar explicitamente um atributo de destino, use a seguinte sintaxe:
[target : attribute-list]
event evento
VALOR DE DESTINO APLICA-SE A
property propriedade
Especifique o valor de destino field para aplicar um atributo ao campo de suporte criado para uma
propriedade autoimplementada.
O exemplo a seguir mostra como aplicar atributos a módulos e assemblies. Para obter mais informações,
consulte Atributos comuns (C#).
using System;
using System.Reflection;
[assembly: AssemblyTitleAttribute("Production assembly 4")]
[module: CLSCompliant(true)]
O exemplo a seguir mostra como aplicar atributos a métodos, parâmetros de método e valores de retorno de
método em C#.
// applies to method
[method: ValidatedContract]
int Method2() { return 0; }
NOTE
Independentemente dos destinos nos quais ValidatedContract é definido como válido, o destino return deve ser
especificado, mesmo se ValidatedContract forem definidos para serem aplicados somente a valores de retorno. Em
outras palavras, o compilador não usará as informações de AttributeUsage para resolver os destinos de atributos
ambíguos. Para obter mais informações, consulte AttributeUsage (C#).
Seções relacionadas
Para obter mais informações, consulte:
Criando atributos personalizados (C#)
Acessando atributos usando reflexão (C#)
Como criar uma C/C++ Union usando atributos ()C#
Atributos comuns (C#)
Informações do chamador (C#)
Consulte também
Guia de Programação em C#
Reflexão (C#)
Atributos
Usando atributos em C#
Criando atributos personalizados (C#)
23/10/2019 • 2 minutes to read • Edit Online
Você pode criar seus próprios atributos personalizados definindo uma classe de atributos, uma classe que deriva
direta ou indiretamente de Attribute, o que faz com que a identificação das definições de atributo nos metadados
seja rápida e fácil. Suponha que você queira marcar tipos com o nome do programador que escreveu o tipo. Você
pode definir uma classe de atributos Author personalizada:
[System.AttributeUsage(System.AttributeTargets.Class |
System.AttributeTargets.Struct)
]
public class Author : System.Attribute
{
private string name;
public double version;
O nome de classe será o nome do atributo, Author . Ela é derivada de System.Attribute , portanto, é uma classe
de atributo personalizado. Os parâmetros do construtor são parâmetros posicionais do atributo personalizado.
Neste exemplo, name é um parâmetro posicional. Quaisquer propriedades ou campos públicos de
leitura/gravação são chamados de parâmetros. Nesse caso, version é o único parâmetro nomeado. Observe o
uso do atributo AttributeUsage para tornar o atributo Author válido apenas na classe e nas declarações struct .
Você pode usar esse novo atributo da seguinte maneira:
AttributeUsage tem um parâmetro nomeado, AllowMultiple , com o qual você pode fazer um atributo
personalizado de uso único ou mulituso. No exemplo de código a seguir, um atributo multiuso é criado.
[System.AttributeUsage(System.AttributeTargets.Class |
System.AttributeTargets.Struct,
AllowMultiple = true) // multiuse attribute
]
public class Author : System.Attribute
No exemplo de código a seguir, vários atributos do mesmo tipo são aplicados a uma classe.
[Author("P. Ackerman", version = 1.1)]
[Author("R. Koch", version = 1.2)]
class SampleClass
{
// P. Ackerman's code goes here...
// R. Koch's code goes here...
}
Consulte também
System.Reflection
Guia de Programação em C#
Escrevendo atributos personalizados
Reflexão (C#)
Atributos (C#)
Acessando atributos usando reflexão (C#)
AttributeUsage (C#)
AttributeUsage (C#)
23/10/2019 • 4 minutes to read • Edit Online
Determina como uma classe de atributo personalizado pode ser usada. AttributeUsageAttribute é um atributo
aplicado a definições de atributo personalizado. O atributo AttributeUsage permite que você controle:
A quais elementos do programa o atributo pode ser aplicado. A menos que você restrinja seu uso, um atributo
poderá ser aplicado a qualquer um dos seguintes elementos do programa:
assembly
module
campo
evento
method
param
propriedade
return
tipo
Indica se um atributo pode ser aplicado a um único elemento do programa várias vezes.
Indica se os atributos são herdados por classes derivadas.
As configurações padrão se parecem com o seguinte exemplo quando aplicadas explicitamente:
[System.AttributeUsage(System.AttributeTargets.All,
AllowMultiple = false,
Inherited = true)]
class NewAttribute : System.Attribute { }
Neste exemplo, a classe NewAttribute pode ser aplicada a qualquer elemento de programa compatível. Porém, ele
pode ser aplicado apenas uma vez para cada entidade. O atributo é herdado por classes derivadas quando
aplicado a uma classe base.
Os argumentos AllowMultiple e Inherited são opcionais e, portanto, o seguinte código tem o mesmo efeito:
[System.AttributeUsage(System.AttributeTargets.All)]
class NewAttribute : System.Attribute { }
[AttributeUsage(AttributeTargets.Property | AttributeTargets.Field)]
class NewPropertyOrFieldAttribute : Attribute { }
Do C# 7.3 em diante, os atributos podem ser aplicados à propriedade ou ao campo de suporte de uma
propriedade autoimplementada. O atributo se aplica à propriedade, a menos que você especifique o especificador
field no atributo. Ambos são mostrados no seguinte exemplo:
class MyClass
{
// Attribute attached to property:
[NewPropertyOrField]
public string Name { get; set; }
Se o argumento AllowMultiple for true , o atributo resultante poderá ser aplicado mais de uma vez a uma única
entidade, conforme mostrado no seguinte exemplo:
[MultiUse]
[MultiUse]
class Class1 { }
[MultiUse, MultiUse]
class Class2 { }
Nesse caso, MultiUseAttribute pode ser aplicado repetidas vezes porque AllowMultiple está definido como
true . Os dois formatos mostrados para a aplicação de vários atributos são válidos.
Se Inherited for false , o atributo não será herdado por classes derivadas de uma classe atribuída. Por exemplo:
[NonInherited]
class BClass { }
Comentários
O atributo AttributeUsage é um atributo de uso único. Ele não pode ser aplicado mais de uma vez à mesma
classe. AttributeUsage é um alias para AttributeUsageAttribute.
Para obter mais informações, consulte Acessando atributos usando reflexão (C#).
Exemplo
O exemplo a seguir demonstra o efeito dos argumentos Inherited e AllowMultiple no atributo
AttributeUsageAttribute e como os atributos personalizados aplicados a uma classe podem ser enumerados.
using System;
[AttributeUsage(AttributeTargets.Class)]
class SecondAttribute : Attribute { }
[Third, Third]
class DerivedClass : BaseClass { }
Saída de Exemplo
Attributes on Base Class:
FirstAttribute
SecondAttribute
Attributes on Derived Class:
ThirdAttribute
ThirdAttribute
SecondAttribute
Consulte também
Attribute
System.Reflection
Guia de Programação em C#
Atributos
Reflexão (C#)
Atributos
Criando atributos personalizados (C#)
Acessando atributos usando reflexão (C#)
Acessando atributos usando reflexão (C#)
23/10/2019 • 3 minutes to read • Edit Online
O fato de que você pode definir atributos personalizados e colocá-los em seu código-fonte seria de pouco valor
sem alguma maneira de recuperar essas informações e tomar ação sobre elas. Por meio de reflexão, você pode
recuperar as informações que foram definidas com atributos personalizados. O método principal é o
GetCustomAttributes , que retorna uma matriz de objetos que são equivalentes, em tempo de execução, aos
atributos do código-fonte. Esse método tem várias versões sobrecarregadas. Para obter mais informações,
consulte Attribute.
Uma especificação de atributo, como:
No entanto, o código não será executado até que SampleClass tenha os atributos consultados. Chamar
GetCustomAttributes na SampleClass faz com que um objeto Author seja criado e inicializado como acima. Se a
classe tiver outros atributos, outros objetos de atributo serão construídos de forma semelhante. Então o
GetCustomAttributes retornará o objeto Author e quaisquer outros objetos de atributo em uma matriz. Você
poderá iterar sobre essa matriz, determinar quais atributos foram aplicados com base no tipo de cada elemento
da matriz e extrair informações dos objetos de atributo.
Exemplo
Aqui está um exemplo completo. Um atributo personalizado é definido, aplicado a várias entidades e recuperado
por meio da reflexão.
// Multiuse attribute.
[System.AttributeUsage(System.AttributeTargets.Class |
System.AttributeTargets.Struct,
AllowMultiple = true) // Multiuse attribute.
]
public class Author : System.Attribute
{
string name;
public double version;
// Default value.
version = 1.0;
}
class TestAuthorAttribute
{
static void Test()
{
PrintAuthorInfo(typeof(FirstClass));
PrintAuthorInfo(typeof(SecondClass));
PrintAuthorInfo(typeof(ThirdClass));
}
// Using reflection.
System.Attribute[] attrs = System.Attribute.GetCustomAttributes(t); // Reflection.
// Displaying output.
foreach (System.Attribute attr in attrs)
{
if (attr is Author)
{
Author a = (Author)attr;
System.Console.WriteLine(" {0}, version {1:f}", a.GetName(), a.version);
}
}
}
}
/* Output:
Author information for FirstClass
P. Ackerman, version 1.00
Author information for SecondClass
Author information for ThirdClass
R. Koch, version 2.00
P. Ackerman, version 1.00
*/
Consulte também
System.Reflection
Attribute
Guia de Programação em C#
Recuperando informações armazenadas em atributos
Reflexão (C#)
Atributos (C#)
Criando atributos personalizados (C#)
Como criar uma C/C++ Union usando atributos ()C#
25/11/2019 • 2 minutes to read • Edit Online
Usando atributos, você pode personalizar a forma como as estruturas são colocadas na memória. Por exemplo,
você pode criar o que é conhecido como uma união no C/C++ usando os atributos
StructLayout(LayoutKind.Explicit) e FieldOffset .
Exemplo
Neste segmento de código, todos os campos de TestUnion são iniciados no mesmo local na memória.
[System.Runtime.InteropServices.StructLayout(LayoutKind.Explicit)]
struct TestUnion
{
[System.Runtime.InteropServices.FieldOffset(0)]
public int i;
[System.Runtime.InteropServices.FieldOffset(0)]
public double d;
[System.Runtime.InteropServices.FieldOffset(0)]
public char c;
[System.Runtime.InteropServices.FieldOffset(0)]
public byte b;
}
Exemplo
A seguir, temos outro exemplo em que os campos são iniciados em locais diferentes definidos explicitamente.
[System.Runtime.InteropServices.StructLayout(LayoutKind.Explicit)]
struct TestExplicit
{
[System.Runtime.InteropServices.FieldOffset(0)]
public long lg;
[System.Runtime.InteropServices.FieldOffset(0)]
public int i1;
[System.Runtime.InteropServices.FieldOffset(4)]
public int i2;
[System.Runtime.InteropServices.FieldOffset(8)]
public double d;
[System.Runtime.InteropServices.FieldOffset(12)]
public char c;
[System.Runtime.InteropServices.FieldOffset(14)]
public byte b;
}
Os dois campos inteiros, i1 e i2 , compartilham os mesmos locais de memória que lg . Esse tipo de controle
sobre o layout do struct é útil ao usar a invocação de plataforma.
Consulte também
System.Reflection
Attribute
Guia de Programação em C#
Atributos
Reflexão (C#)
Atributos (C#)
Criando atributos personalizados (C#)
Acessando atributos usando reflexão (C#)
Atributos comuns (C#)
23/10/2019 • 12 minutes to read • Edit Online
Este tópico descreve os atributos que são mais comumente usados nos programas em C#.
Atributos globais
Atributo obsoleto
Atributo condicional
Atributos de informações do chamador
Atributos globais
A maioria dos atributos são aplicados aos elementos específicos de linguagem, como classes ou métodos. No
entanto, alguns atributos são globais. Eles se aplicam a um assembly inteiro ou módulo. Por exemplo, o atributo
AssemblyVersionAttribute pode ser usado para inserir informações de versão em um assembly, desta maneira:
[assembly: AssemblyVersion("1.0.0.0")]
Os atributos globais aparecem no código-fonte depois de qualquer diretiva using de nível superior e antes de
qualquer declaração de namespace, módulo ou tipo. Os atributos globais podem aparecer em vários arquivos de
origem, mas os arquivos devem ser compilados em uma única passagem de compilação. Em projetos do C#, os
atributos globais são colocados no arquivo AssemblyInfo.cs.
Os atributos de assembly são valores que fornecem informações sobre um assembly. Eles se enquadram nas
seguintes categorias:
Atributos de identidade do assembly
Atributos informativos
Atributos de manifesto do assembly
Atributos de Identidade do Assembly
Três atributos (com um nome forte, se aplicável) determinam a identidade de um assembly: nome, versão e
cultura. Esses atributos formam o nome completo do assembly e são necessários ao fazer referência a ele no
código. Você pode definir a versão e a cultura de um assembly, usando atributos. No entanto, o valor do nome é
definido pelo compilador, pelo IDE do Visual Studio na caixa de diálogo de Informações do Assembly ou pelo
Assembly Linker (Al.exe) quando o assembly é criado, com base no arquivo que contém o manifesto do assembly.
O atributo AssemblyFlagsAttribute especifica se várias cópias do assembly podem coexistir.
A tabela a seguir mostra os atributos de identidade.
ATRIBUTO FINALIDADE
Atributos Informativos
Você pode usar atributos informativos para fornecer informações adicionais corporativas ou de produto para um
assembly. A tabela a seguir mostra os atributos informativos definidos no namespace System.Reflection.
ATRIBUTO FINALIDADE
ATRIBUTO FINALIDADE
Atributo obsoleto
O atributo Obsolete marca uma entidade programa como não recomendada para uso. Cada uso de uma
entidade marcada como obsoleta gerará subsequentemente um aviso ou erro, dependendo de como o atributo é
configurado. Por exemplo:
Neste exemplo o atributo Obsolete é aplicado à classe A e ao método B.OldMethod . Como o segundo
argumento do construtor de atributo aplicado a B.OldMethod está definido como true , esse método causará um
erro do compilador, enquanto que ao usar a classe A , produzirá apenas um aviso. Chamar B.NewMethod , no
entanto, não produz aviso nem erro.
A cadeia de caracteres fornecida como o primeiro argumento para o construtor de atributo será exibida como
parte do aviso ou erro. Por exemplo, ao usá-lo com as definições anteriores, o código a seguir gera um erro e dois
avisos:
// Generates 2 warnings:
// A a = new A();
São gerados dois avisos para a classe A : um para a declaração da referência de classe e outro para o construtor
de classe.
O atributo Obsolete pode ser usado sem argumentos, mas é recomendável incluir uma explicação de por que o
item está obsoleto e o que deve ser usado no lugar dele.
O atributo Obsolete é um atributo de uso único e pode ser aplicado a qualquer entidade que permite atributos.
Obsolete é um alias para ObsoleteAttribute.
Atributo condicional
O atributo Conditional torna a execução de um método dependente de um identificador de pré-processamento.
O atributo Conditional é um alias para ConditionalAttribute e pode ser aplicado a um método ou uma classe de
atributo.
Neste exemplo, Conditional é aplicado a um método para habilitar ou desabilitar a exibição de informações de
diagnóstico específicas do programa:
#define TRACE_ON
using System;
using System.Diagnostics;
Se o identificador TRACE_ON não estiver definido, nenhuma saída de rastreamento será exibida.
O atributo Conditional é frequentemente usado com o identificador DEBUG para habilitar recursos de
rastreamento e de registro em log para builds de depuração, mas não em builds de versão, dessa maneira:
[Conditional("DEBUG")]
static void DebugMethod()
{
}
Quando um método marcado como condicional é chamado, a presença ou ausência do símbolo de pré-
processamento especificado determina se a chamada será incluída ou omitida. Se o símbolo estiver definido, a
chamada será incluída, caso contrário, a chamada será omitida. O uso de Conditional é uma alternativa mais
limpa, mais elegante e menos propensa a erros para incluir métodos dentro de blocos #if…#endif , dessa maneira:
#if DEBUG
void ConditionalMethod()
{
}
#endif
Um método condicional deve ser um método em uma declaração de classe ou de struct e não deve ter um valor
retornado.
Usando vários identificadores
Se um método tem vários atributos Conditional , uma chamada para o método será incluída se pelo menos um
dos símbolos condicionais for definido (em outras palavras, os símbolos são logicamente vinculados através do
uso do operador OR ). Neste exemplo, a presença de A ou B resultará em uma chamada de método:
[Conditional("A"), Conditional("B")]
static void DoIfAorB()
{
// ...
}
Para alcançar o efeito da vinculação lógica de símbolos usando o operador AND, você pode definir métodos
condicionais em série. Por exemplo, o segundo método abaixo será executado somente se A e B estiverem
definidos:
[Conditional("A")]
static void DoIfA()
{
DoIfAandB();
}
[Conditional("B")]
static void DoIfAandB()
{
// Code to execute when both A and B are defined...
}
[Conditional("DEBUG")]
public class Documentation : System.Attribute
{
string text;
class SampleClass
{
// This attribute will only be included if DEBUG is defined.
[Documentation("This method displays an integer.")]
static void DoWork(int i)
{
System.Console.WriteLine(i.ToString());
}
}
Para obter mais informações sobre os atributos de informações do chamador, consulte Informações do chamador
(C#).
Consulte também
System.Reflection
Attribute
Guia de Programação em C#
Atributos
Reflexão (C#)
Acessando atributos usando reflexão (C#)
Informações do chamador (C#)
23/10/2019 • 5 minutes to read • Edit Online
Ao usar atributos de informações do chamador, você pode obter informações sobre o chamador de um método.
Você pode obter o caminho do arquivo do código-fonte, o número da linha no código-fonte e o nome do
membro do chamador. Essas informações são úteis para fins de rastreamento, depuração e criação de
ferramentas de diagnóstico.
Para obter essas informações, você deve usar os atributos que são aplicadas aos parâmetros opcionais, cada qual
com um valor padrão. A tabela a seguir lista os atributos de informações do chamador que são definidos no
namespace de System.Runtime.CompilerServices:
Exemplo
O exemplo a seguir mostra como usar os atributos de informações do chamador. Em cada chamada para o
método TraceMessage , as informações do chamador são substituídas como argumentos para os parâmetros
opcionais.
// Sample Output:
// message: Something happened.
// member name: DoProcessing
// source file path: c:\Visual Studio Projects\CallerInfoCS\CallerInfoCS\Form1.cs
// source line number: 31
Comentários
Você deve especificar um valor padrão explícito para cada parâmetro opcional. Você não pode aplicar atributos
de informações do chamador aos parâmetros que não são especificados como opcionais.
Os atributos de informações do chamador não tornam um parâmetro opcional. Em vez disso, eles afetam o valor
padrão que é passado quando o argumento é omitido.
Os valores de informações do chamador são emitidos como literais em linguagem intermediária (IL ) em tempo
de compilação. Ao contrário dos resultados da propriedade StackTrace para exceções, os resultados não são
afetados por ofuscamento.
Você pode fornecer explicitamente os argumentos opcionais para controlar as informações do chamador ou
ocultá-las.
Nomes dos membros
Você pode usar o atributo CallerMemberName para evitar especificar o nome do membro como um argumento
String ao método chamado. Ao usar essa técnica, você evita o problema de que a Refatoração de
Renomeação não altera os valores de String . Esse benefício é especialmente útil para as seguintes tarefas:
Usar rotinas de rastreamento e diagnóstico.
Implementando a interface INotifyPropertyChanged ao associar dados. Essa interface permite que a
propriedade de um objeto notifique um controle associado sobre a alteração da propriedade de modo que
o controle possa exibir as informações atualizadas. Sem o atributo CallerMemberName , você deve
especificar o nome da propriedade como um literal.
O gráfico a seguir mostra os nomes de membros que são retornados quando você usa o atributo
CallerMemberName .
Operadores usuário ou conversões definidos pelo usuário O nome gerado para o membro, por exemplo, “op_Addition”.
Nenhum membro contentor (por exemplo, nível de assembly O valor padrão do parâmetro opcional.
ou atributos que são aplicadas aos tipos)
Consulte também
Atributos (C#)
Atributos comuns (C#)
Argumentos nomeados e opcionais
Conceitos de programação (C#)
Coleções (C#)
04/11/2019 • 21 minutes to read • Edit Online
Para muitos aplicativos, você desejará criar e gerenciar grupos de objetos relacionados. Há duas maneiras de
agrupar objetos: criando matrizes de objetos e criando coleções de objetos.
As matrizes são mais úteis ao criar e trabalhar com um número fixo de objetos fortemente tipados. Para obter
informações sobre matrizes, consulte Matrizes.
As coleções fornecem uma maneira mais flexível de trabalhar com grupos de objetos. Ao contrário das matrizes, o
grupo de objetos com o qual você trabalha pode crescer e reduzir dinamicamente conforme as necessidades do
aplicativo são alteradas. Para algumas coleções, você pode atribuir uma chave para qualquer objeto que colocar na
coleção para que você possa recuperar rapidamente o objeto, usando a chave.
Uma coleção é uma classe, portanto você deve declarar uma instância da classe antes de adicionar elementos a
essa coleção.
Se a coleção contiver elementos de apenas um tipo de dados, você poderá usar uma das classes no namespace
System.Collections.Generic. Uma coleção genérica impõe segurança de tipos para que nenhum outro tipo de
dados possa ser adicionado a ela. Ao recuperar um elemento de uma coleção genérica, você não precisa
determinar seu tipo de dados ou convertê-lo.
NOTE
Para os exemplos neste tópico, inclua diretivas using para os namespaces System.Collections.Generic e System.Linq .
Neste tópico
Usando uma coleção simples
Tipos de coleções
Classes System.Collections.Generic
Classes System.Collections.Concurrent
Classes System.Collections
Implementando uma coleção de pares chave-valor
Usando LINQ para acessar uma coleção
Classificando uma coleção
Definindo uma coleção personalizada
Iteradores
Se o conteúdo de uma coleção for conhecido com antecedência, você poderá usar um inicializador de coleção
para inicializar a coleção. Para obter mais informações, consulte Inicializadores de coleção e objeto.
O exemplo a seguir é igual ao exemplo anterior, exceto que um inicializador de coleção é usado para adicionar
elementos à coleção.
Você pode usar uma instrução for em vez de uma instrução foreach para iterar em uma coleção. Você realiza isso
acessando os elementos da coleção pela posição do índice. O índice dos elementos começa em 0 e termina na
contagem de elementos, menos de 1.
O exemplo a seguir itera nos elementos de uma coleção usando for em vez de foreach .
O exemplo a seguir remove elementos de uma lista genérica. Em vez de uma instrução foreach , é usada uma
instrução for que itera em ordem decrescente. Isso é feito porque o método RemoveAt faz com que os elementos
após um elemento removido tenham um valor de índice menor.
Para o tipo dos elementos na List<T>, você também pode definir sua própria classe. No exemplo a seguir, a classe
Galaxy que é usada pela List<T> é definida no código.
private static void IterateThroughList()
{
var theGalaxies = new List<Galaxy>
{
new Galaxy() { Name="Tadpole", MegaLightYears=400},
new Galaxy() { Name="Pinwheel", MegaLightYears=25},
new Galaxy() { Name="Milky Way", MegaLightYears=0},
new Galaxy() { Name="Andromeda", MegaLightYears=3}
};
// Output:
// Tadpole 400
// Pinwheel 25
// Milky Way 0
// Andromeda 3
}
Tipos de coleções
Várias coleções comuns são fornecidas pelo .NET Framework. Cada tipo de coleção é projetado para uma
finalidade específica.
Algumas das classes de coleção comuns são descritas nesta seção:
Classes System.Collections.Generic
Classes System.Collections.Concurrent
Classes System.Collections
Classes System.Collections.Generic
Você pode criar uma coleção genérica usando uma das classes no namespace System.Collections.Generic. Uma
coleção genérica é útil quando cada item na coleção tem o mesmo tipo de dados. Uma coleção genérica impõe
tipagem forte, permitindo que apenas o tipo de dados desejado seja adicionado.
A tabela a seguir lista algumas das classes frequentemente usadas do namespace System.Collections.Generic:
CLASS DESCRIÇÃO
List<T> Representa uma lista de objetos que podem ser acessados por
índice. Fornece métodos para pesquisar, classificar e modificar
listas.
Para obter informações adicionais, consulte Tipos de coleção comumente usados, Selecionando uma classe de
coleção e System.Collections.Generic.
Classes System.Collections.Concurrent
No .NET Framework 4 ou mais recente, as coleções no namespace System.Collections.Concurrent fornecem
operações thread-safe eficientes para acessar itens da coleção de vários threads.
As classes no namespace System.Collections.Concurrent deverão ser usadas em vez dos tipos correspondentes
nos namespaces System.Collections.Generic e System.Collections sempre que vários threads estiverem acessando
a coleção simultaneamente. Para obter mais informações, veja Coleções thread-safe e
System.Collections.Concurrent.
Algumas classes incluídas no namespace System.Collections.Concurrent são BlockingCollection<T>,
ConcurrentDictionary<TKey,TValue>, ConcurrentQueue<T> e ConcurrentStack<T>.
Classes System.Collections
As classes no namespace System.Collections não armazenam elementos como objetos especificamente tipados,
mas como objetos do tipo Object .
Sempre que possível, você deve usar as coleções genéricas no namespace System.Collections.Generic ou no
System.Collections.Concurrent em vez dos tipos herdados no namespace System.Collections .
A tabela a seguir lista algumas das classes frequentemente usadas no namespace System.Collections :
CLASS DESCRIÇÃO
return elements;
}
theElement.Symbol = symbol;
theElement.Name = name;
theElement.AtomicNumber = atomicNumber;
Para, em vez disso, usar um inicializador de coleção para criar a coleção Dictionary , você pode substituir os
métodos BuildDictionary e AddToDictionary pelo seguinte método.
if (elements.ContainsKey(symbol) == false)
{
Console.WriteLine(symbol + " not found");
}
else
{
Element theElement = elements[symbol];
Console.WriteLine("found: " + theElement.Name);
}
}
O exemplo a seguir usa o método TryGetValue para localizar rapidamente um item por chave.
// LINQ Query.
var subset = from theElement in elements
where theElement.AtomicNumber < 22
orderby theElement.Name
select theElement;
// Output:
// Calcium 20
// Potassium 19
// Scandium 21
}
// Output:
// blue 50 car4
// blue 30 car5
// blue 20 car1
// green 50 car7
// green 10 car3
// red 60 car6
// red 50 car2
}
return compare;
}
}
// Collection class.
public class AllColors : System.Collections.IEnumerable
{
Color[] _colors =
{
new Color() { Name = "red" },
new Color() { Name = "blue" },
new Color() { Name = "green" }
};
// Custom enumerator.
private class ColorEnumerator : System.Collections.IEnumerator
{
private Color[] _colors;
private int _position = -1;
object System.Collections.IEnumerator.Current
{
get
{
return _colors[_position];
}
}
bool System.Collections.IEnumerator.MoveNext()
{
_position++;
return (_position < _colors.Length);
}
void System.Collections.IEnumerator.Reset()
{
_position = -1;
}
}
}
}
// Element class.
public class Color
{
public string Name { get; set; }
}
Iterators
Um iterador é usado para realizar uma iteração personalizada em uma coleção. Um iterador pode ser um método
ou um acessador get . Um iterador usa uma instrução yield return para retornar um elemento da coleção por vez.
Você chama um iterador usando uma instrução foreach. Cada iteração do loop foreach chama o iterador. Quando
uma instrução yield return é alcançada no iterador, uma expressão é retornada e o local atual no código é retido.
A execução será reiniciada desse local na próxima vez que o iterador for chamado.
Para obter mais informações, consulte Iteradores (C#).
O exemplo a seguir usa um método iterador. O método iterador tem uma instrução yield return que está dentro
de um loop for. No método ListEvenNumbers , cada iteração do corpo da instrução foreach cria uma chamada ao
método iterador, que avança para a próxima instrução yield return .
Consulte também
Inicializadores de objeto e coleção
Conceitos de programação (C#)
Instrução Option Strict
LINQ to Objects (C#)
PLINQ (LINQ paralelo)
Coleções e Estruturas de Dados
Selecionando uma Classe de Coleção
Comparações e Classificações Dentro de Coleções
Quando Usar Coleções Genéricas
Covariância e contravariância (C#)
23/10/2019 • 5 minutes to read • Edit Online
No C#, a covariância e a contravariância habilitam a conversão de referência implícita para tipos de matriz, tipos
de delegados e argumentos de tipo genérico. A covariância preserva a compatibilidade de atribuição, e a
contravariância reverte.
O código a seguir demonstra a diferença entre a compatibilidade da atribuição, a covariância e a contravariância.
// Assignment compatibility.
string str = "test";
// An object of a more derived type is assigned to an object of a less derived type.
object obj = str;
// Covariance.
IEnumerable<string> strings = new List<string>();
// An object that is instantiated with a more derived type argument
// is assigned to an object instantiated with a less derived type argument.
// Assignment compatibility is preserved.
IEnumerable<object> objects = strings;
// Contravariance.
// Assume that the following method is in the class:
// static void SetObject(object o) { }
Action<object> actObject = SetObject;
// An object that is instantiated with a less derived type argument
// is assigned to an object instantiated with a more derived type argument.
// Assignment compatibility is reversed.
Action<string> actString = actObject;
A covariância para matrizes permite a conversão implícita de uma matriz de um tipo mais derivado para uma
matriz de um tipo menos derivado. Mas essa operação não é fortemente tipada, conforme mostrado no exemplo
de código a seguir.
Uma interface ou delegado genérico será chamado variante se seus parâmetros genéricos forem declarados
covariantes ou contravariantes. O C# permite que você crie suas próprias interfaces variantes e delegados. Para
obter mais informações, consulte Criando interfaces genéricas variantes (C#) e Variação em delegados (C#).
Tópicos relacionados
TÍTULO DESCRIÇÃO
Criando interfaces genéricas variáveis (C#) Mostra como criar interfaces variantes personalizadas.
Usando variação em interfaces para coleções genéricas (C#) Mostra como o suporte de covariância e contravariância nas
interfaces IEnumerable<T> e IComparable<T> pode ajudar
na reutilização do código.
Usando variação em delegados (C#) Mostra como usar o suporte de covariância e contravariância
em delegados não genéricos para corresponder às assinaturas
de método com tipos delegados.
Usando variação para delegados genéricos Func e Action (C#) Mostra como o suporte de covariância e contravariância nos
delegados Func e Action pode ajudar na reutilização do
código.
Variância em interfaces genéricas (C#)
23/10/2019 • 4 minutes to read • Edit Online
O .NET Framework 4 introduziu o suporte à variação para diversas interfaces genéricas existentes. O suporte à
variação possibilita a conversão implícita de classes que implementam essas interfaces.
A partir do .NET Framework 4, as seguintes interfaces são variantes:
IEnumerable<T> (T é covariante)
IEnumerator<T> (T é covariante)
IQueryable<T> (T é covariante)
IGrouping<TKey,TElement> ( TKey e TElement são covariantes)
IComparer<T> (T é contravariante)
IEqualityComparer<T> (T é contravariante)
IComparable<T> (T é contravariante)
A partir do .NET Framework 4.5, as seguintes interfaces são variantes:
IReadOnlyList<T> (T é covariante)
IReadOnlyCollection<T> (T é covariante)
A covariância permite que um método tenha um tipo de retorno mais derivados que aquele definidos pelo
parâmetro de tipo genérico da interface. Para ilustrar o recurso de covariância, considere estas interfaces
genéricas: IEnumerable<Object> e IEnumerable<String> . A interface IEnumerable<String> não herda a interface
IEnumerable<Object> . No entanto, o tipo String herda o tipo Object e, em alguns casos, talvez você queira
atribuir objetos dessas interfaces uns aos outros. Isso será mostrado no exemplo de código a seguir.
Em versões anteriores do .NET Framework, esse código causa um erro de compilação em C# e, se Option Strict
estiver ativado, no Visual Basic. Mas agora você pode usar strings em vez de objects , conforme mostrado no
exemplo anterior, porque a interface IEnumerable<T> é covariante.
A contravariância permite que um método tenha tipos de argumentos menos derivados que aquele especificado
pelo parâmetro genérico da interface. Para ilustrar a contravariância, suponha que você tenha criado uma classe
BaseComparer para comparar instâncias da classe BaseClass . A classe BaseComparer implementa a interface
IEqualityComparer<BaseClass> . Como a interface IEqualityComparer<T> agora é contravariante, você pode usar
BaseComparer para comparar instâncias de classes que herdam a classe BaseClass . Isso será mostrado no
exemplo de código a seguir.
// Simple hierarchy of classes.
class BaseClass { }
class DerivedClass : BaseClass { }
// Comparer class.
class BaseComparer : IEqualityComparer<BaseClass>
{
public int GetHashCode(BaseClass baseInstance)
{
return baseInstance.GetHashCode();
}
public bool Equals(BaseClass x, BaseClass y)
{
return x == y;
}
}
class Program
{
static void Test()
{
IEqualityComparer<BaseClass> baseComparer = new BaseComparer();
Para ver mais exemplos, consulte Usando variação em interfaces para coleções genéricas (C#).
A variação em interfaces genéricas tem suporte somente para tipos de referência. Tipos de valor não dão suporte
à variação. Por exemplo, IEnumerable<int> não pode ser convertido implicitamente em IEnumerable<object> ,
porque inteiros são representados por um tipo de valor.
Também é importante lembrar que as classes que implementam interfaces variantes ainda são invariantes. Por
exemplo, embora List<T> implemente a interface covariante IEnumerable<T>, você não pode converter
implicitamente List<String> para List<Object> . Isso é ilustrado no exemplo de código a seguir.
Consulte também
Usando variação em interfaces para coleções genéricas (C#)
Criando interfaces genéricas variáveis (C#)
Interfaces genéricas
Variação em delegados (C#)
Criando interfaces genéricas variantes (C#)
31/10/2019 • 8 minutes to read • Edit Online
Você pode declarar parâmetros de tipo genérico em interfaces como covariantes ou contravariantes. A
Covariância permite que os métodos de interface tenham tipos de retorno mais derivados que aqueles definidos
pelos parâmetros de tipo genérico. A Contravariância permite que os métodos de interface tenham tipos de
argumentos que são menos derivados que aqueles especificados pelos parâmetros genéricos. Uma interface
genérica que tenha parâmetros de tipo genérico covariantes ou contravariantes é chamada de variante.
NOTE
O .NET Framework 4 introduziu o suporte à variação para diversas interfaces genéricas existentes. Para obter a lista das
interfaces de variantes no .NET Framework, consulte Variância em interfaces genéricas (C#).
IMPORTANT
Os parâmetros ref , in e out no C# não podem ser variantes. Os tipos de valor também não dão suporte à variância.
Você pode declarar um parâmetro de tipo genérico como covariante usando a palavra-chave out . O tipo de
covariante deve satisfazer as condições a seguir:
O tipo é usado apenas como um tipo de retorno dos métodos de interface e não é usado como um tipo de
argumentos de método. Isso é ilustrado no exemplo a seguir, no qual o tipo R é declarado covariante.
Há uma exceção a essa regra. Se você tiver um delegado genérico contravariante como um parâmetro de
método, você poderá usar o tipo como um parâmetro de tipo genérico para o delegado. Isso é ilustrado
pelo tipo R no exemplo a seguir. Para obter mais informações, consulte Variância em delegados (C#) e
Usando variância para delegados genéricos Func e Action (C#).
O tipo não é usado como uma restrição genérica para os métodos de interface. O código a seguir ilustra
isso.
interface ICovariant<out R>
{
// The following statement generates a compiler error
// because you can use only contravariant or invariant types
// in generic constraints.
// void DoSomething<T>() where T : R;
}
Você pode declarar um parâmetro de tipo genérico como contravariante usando a palavra-chave in . O tipo
contravariante pode ser usado apenas como um tipo de argumentos de método e não como um tipo de retorno
dos métodos de interface. O tipo contravariante também pode ser usado para restrições genéricas. O código a
seguir mostra como declarar uma interface contravariante e usar uma restrição genérica para um de seus
métodos.
Também é possível oferecer suporte à covariância e contravariância na mesma interface, mas para parâmetros de
tipo diferentes, conforme mostrado no exemplo de código a seguir.
As classes que implementam interfaces variantes são invariantes. Por exemplo, considere o código a seguir.
// The interface is covariant.
ICovariant<Button> ibutton = new SampleImplementation<Button>();
ICovariant<Object> iobj = ibutton;
Na interface IInvariant<T> , o parâmetro de tipo genérico T é invariante, enquanto que no IExtCovariant<out T>
o parâmetro de tipo é covariante, embora as duas interfaces estendam a mesma interface. A mesma regra é
aplicada aos parâmetros de tipo genérico contravariantes.
Você pode criar uma interface que estende tanto a interface em que o parâmetro de tipo genérico T é covariante,
quanto a interface em que ele é contravariante, caso na interface de extensão o parâmetro de tipo genérico T seja
invariante. Isso é ilustrado no exemplo de código a seguir.
No entanto, se um parâmetro de tipo genérico T for declarado covariante em uma interface, você não poderá
declará-lo contravariante na interface de extensão ou vice-versa. Isso é ilustrado no exemplo de código a seguir.
Evitando ambiguidade
Ao implementar interfaces genéricas variantes, a variância, às vezes, pode levar à ambiguidade. Isso deve ser
evitado.
Por exemplo, se você implementar explicitamente a mesma interface genérica variante com parâmetros de tipo
genérico diferentes em uma classe, isso poderá criar ambiguidade. O compilador não produzirá um erro nesse
caso, mas não está especificado qual implementação de interface será escolhida em tempo de execução. Isso
poderá causar bugs sutis no seu código. Considere o exemplo de código a seguir.
// Simple class hierarchy.
class Animal { }
class Cat : Animal { }
class Dog : Animal { }
IEnumerator IEnumerable.GetEnumerator()
{
// Some code.
return null;
}
IEnumerator<Dog> IEnumerable<Dog>.GetEnumerator()
{
Console.WriteLine("Dog");
// Some code.
return null;
}
}
class Program
{
public static void Test()
{
IEnumerable<Animal> pets = new Pets();
pets.GetEnumerator();
}
}
Neste exemplo, não está especificado como o método pets.GetEnumerator escolherá entre Cat e Dog . Isso
poderá causar problemas em seu código.
Consulte também
Variância em interfaces genéricas (C#)
Usando variação para delegados genéricos Func e Action (C#)
Usando variação em interfaces para Coleções
Genéricas (C#)
23/10/2019 • 3 minutes to read • Edit Online
Uma interface de covariante permite que seus métodos retornem mais tipos derivados daquelas especificadas na
interface. Uma interface de contravariante permite que seus métodos aceitem parâmetros de tipos menos
derivados do que os especificados na interface.
No .NET Framework 4, várias interfaces existentes se tornaram covariantes e contravariantes. Eles incluem
IEnumerable<T> e IComparable<T>. Isso permite que você reutilize métodos que operam com coleções
genéricas de tipos base para coleções de tipos derivados.
Para obter uma lista de interfaces variantes no .NET Framework, consulte Variação em interfaces genéricas (C#).
class Program
{
// The method has a parameter of the IEnumerable<Person> type.
public static void PrintFullName(IEnumerable<Person> persons)
{
foreach (Person person in persons)
{
Console.WriteLine("Name: {0} {1}",
person.FirstName, person.LastName);
}
}
PrintFullName(employees);
}
}
class Program
{
IEnumerable<Employee> noduplicates =
employees.Distinct<Employee>(new PersonComparer());
Consulte também
Variância em interfaces genéricas (C#)
Variação em delegados (C#)
24/10/2019 • 9 minutes to read • Edit Online
O .NET Framework 3.5 introduziu o suporte a variação para assinaturas de método correspondentes com tipos
de delegados em todos os delegados do C#. Isso significa que você pode atribuir a delegados não apenas os
métodos que têm assinaturas correspondentes, mas também métodos que retornam tipos mais derivados
(covariância) ou que aceitam parâmetros que têm tipos menos derivados (contravariância) do que o
especificado pelo tipo de delegado. Isso inclui delegados genéricos e não genéricos.
Por exemplo, considere o código a seguir, que tem duas classes e dois delegados: genérico e não genérico.
Quando cria delegados dos tipos SampleDelegate ou SampleGenericDelegate<A, R> , você pode atribuir qualquer
um dos seguintes métodos a esses delegados.
// Matching signature.
public static First ASecondRFirst(Second second)
{ return new First(); }
O exemplo de código a seguir ilustra a conversão implícita entre a assinatura do método e o tipo de delegado.
Para obter mais informações, consulte Usando variação em delegados (C#) e Usando variação para os
delegados genéricos Func e Action (C#).
Se você usar somente o suporte à para fazer a correspondência de assinaturas de método com tipos de
delegados e não usar as palavras-chave in e out , você poderá perceber que, às vezes, é possível instanciar
delegados com métodos ou expressões lambda idênticas, mas não é possível atribuir um delegado a outro.
No exemplo de código a seguir, SampleGenericDelegate<String> não pode ser convertido explicitamente em
SampleGenericDelegate<Object> , embora String herde Object . Você pode corrigir esse problema marcando o
parâmetro genérico T com a palavra-chave out .
Você pode declarar um parâmetro de tipo genérico contravariante em um delegado genérico usando a palavra-
chave in . O tipo contravariante pode ser usado apenas como um tipo de argumentos de método e não como
um tipo de retorno de método. O exemplo de código a seguir mostra como declarar um delegado genérico
contravariante.
IMPORTANT
Os parâmetros ref , in e out em C# não podem ser marcados como variantes.
Também é possível dar suporte à variância e à covariância no mesmo delegado, mas para parâmetros de tipo
diferente. Isso é mostrado no exemplo a seguir.
Consulte também
Genéricos
Usando variação para delegados genéricos Func e Action (C#)
Como combinar delegados (delegados multicast)
Usando variação em delegações (C#)
23/10/2019 • 3 minutes to read • Edit Online
Quando você atribui um método a um delegado, a covariância e a contravariância fornece flexibilidade para
corresponder um tipo de delegado a uma assinatura de método. A covariância permite que um método tenha o
tipo de retorno mais derivado do que o definido no delegado. A contravariância permite que um método que
tem tipos de parâmetro menos derivados do que no tipo delegado.
Exemplo 1: Covariância
DESCRIÇÃO
Este exemplo demonstra como delegados podem ser usados com métodos que têm tipos de retorno que são
derivados do tipo de retorno na assinatura do delegado. O tipo de dados retornado por DogsHandler é do tipo
Dogs , que deriva do tipo Mammals definido no delegado.
Código
class Mammals {}
class Dogs : Mammals {}
class Program
{
// Define the delegate.
public delegate Mammals HandlerMethod();
Exemplo 2: Contravariância
DESCRIÇÃO
Este exemplo demonstra como representantes podem ser usados com métodos que têm parâmetros cujos tipos
são tipos base do tipo de parâmetro de assinatura do representante. Com a contravariância, você pode usar um
manipulador de eventos em vez de manipuladores separados. O seguinte exemplo usa dois representantes:
Um representante KeyEventHandler que define a assinatura do evento Button.KeyDown. Sua assinatura é:
public delegate void KeyEventHandler(object sender, KeyEventArgs e)
O exemplo define um manipulador de eventos com um parâmetro EventArgs e o usa para manipular os eventos
Button.KeyDown e Button.MouseClick . Ele pode fazer isso porque EventArgs é um tipo base de KeyEventArgs e
MouseEventArgs.
Código
public Form1()
{
InitializeComponent();
Consulte também
Variação em delegados (C#)
Usando variação para delegados genéricos Func e Action (C#)
Usando variância para delegados genéricos Func e
Action (C#)
23/10/2019 • 2 minutes to read • Edit Online
Esses exemplos demonstram como usar covariância e contravariância nos delegados genéricos Func e Action
para permitir a reutilização dos métodos e fornecer mais flexibilidade em seu código.
Para obter mais informações sobre covariância e contravariância, consulte Variação em delegações (C#).
}
}
Consulte também
Covariância e contravariância (C#)
Genéricos
Árvores de expressão (C#)
25/11/2019 • 7 minutes to read • Edit Online
Árvores de expressão representam código em uma estrutura de dados de árvore, onde cada nó é, por exemplo,
uma expressão, uma chamada de método ou uma operação binária como x < y .
Você pode compilar e executar código representado por árvores de expressão. Isso permite a modificação
dinâmica de código executável, a execução de consultas LINQ em vários bancos de dados e a criação de
consultas dinâmicas. Para obter mais informações sobre árvores de expressão no LINQ, consulte como usar
árvores de expressão para criar consultasC#dinâmicas ().
Árvores de expressão também são usadas no runtime de linguagem dinâmica (DLR ) para fornecer
interoperabilidade entre linguagens dinâmicas e o .NET Framework e permitir que gravadores compiladores
emitam árvores de expressão em vez de Microsoft intermediate language (MSIL ). Para obter mais informações
sobre o DLR, consulte Visão geral do Dynamic Language Runtime.
Você pode fazer o compilador C# ou do Visual Basic criar uma árvore de expressões para você com base em
uma expressão lambda anônima ou criar árvores de expressão manualmente usando o namespace
System.Linq.Expressions.
Console.WriteLine(factorial);
// Prints 120.
Para saber mais, confira Gerar métodos dinâmicos com árvores de expressão no Visual Studio 2010, que
também se aplica a versões mais recentes do Visual Studio.
// Prints True.
Consulte também
System.Linq.Expressions
Como executar árvores de expressão (C#)
Como modificar árvores de expressão (C#)
Expressões Lambda
Visão geral do Dynamic Language Runtime
Conceitos de programação (C#)
Como executar árvores de expressão (C#)
25/11/2019 • 2 minutes to read • Edit Online
Este tópico mostra como executar uma árvore de expressão. Executar uma árvore de expressão pode retornar um
valor ou apenas realizar uma ação, como chamar um método.
Somente árvores de expressão que representam expressões lambda podem ser executadas. Árvores de expressão
que representam expressões lambda são do tipo LambdaExpression ou Expression<TDelegate>. Para executar
essas árvores de expressão, chame o método Compile para criar um delegado executável e, em seguida, invoque
o delegado.
NOTE
Se o tipo de delegado não for conhecido, ou seja, a expressão lambda for do tipo LambdaExpression e não
Expression<TDelegate>, você deverá chamar o método DynamicInvoke sobre o delegado em vez de invocá-la diretamente.
Se uma árvore de expressão não representa uma expressão lambda, você pode criar uma nova expressão lambda
que tenha a árvore de expressão original como corpo, chamando o método Lambda<TDelegate>(Expression,
IEnumerable<ParameterExpression>). Em seguida, você pode executar a expressão lambda como descrito
anteriormente nesta seção.
Exemplo
O exemplo de código a seguir demonstra como executar uma árvore de expressão que representa a elevação de
um número a uma potência, criando uma expressão lambda e executando-a. O resultado, representado pelo
número elevado à potência, é exibido.
Compilando o código
Inclua o namespace System.Linq.Expressions.
Consulte também
Árvores de expressão (C#)
Como modificar árvores de expressão (C#)
Como modificar árvores de expressão (C#)
25/11/2019 • 3 minutes to read • Edit Online
Este tópico mostra como modificar uma árvore de expressão. As árvores de expressão são imutáveis, o que
significa que elas não podem ser diretamente modificadas. Para alterar uma árvore de expressão, você deve criar
uma cópia de uma árvore de expressão existente e, ao criar a cópia, faça as alterações necessárias. Você pode usar
a classe ExpressionVisitor para percorrer uma árvore de expressão existente e copiar cada nó que ela visitar.
Para modificar uma árvore de expressão
1. Crie um novo projeto de Aplicativo de Console.
2. Adicione uma diretiva using ao arquivo para o namespace System.Linq.Expressions .
3. Adicione a classe AndAlsoModifier ao seu projeto.
return base.VisitBinary(b);
}
}
Essa classe herda a classe ExpressionVisitor e é especializada para modificar expressões que representam
operações AND condicionais. Ela muda essas operações de uma AND condicional para uma OR
condicional. Para fazer isso, a classe substitui o método VisitBinary do tipo base, pois as expressões AND
condicionais são representadas como expressões binárias. No método VisitBinary , se a expressão que é
passada a ele representa uma operação AND condicional, o código cria uma nova expressão que contém o
operador OR condicional em vez do operador AND condicional. Se a expressão que é passada para o
VisitBinary não representa uma operação AND condicional, o método adia para a implementação da
classe base. Os métodos da classe base constroem nós que são semelhantes às árvores de expressão que
são passadas, mas os nós têm suas subárvores substituídas pelas árvores de expressão que são produzidas
recursivamente pelo visitante.
4. Adicione uma diretiva using ao arquivo para o namespace System.Linq.Expressions .
5. Adicione código para o método Main no arquivo Program.cs para criar uma árvore de expressão e passá-
la ao método que a modificará.
Expression<Func<string, bool>> expr = name => name.Length > 10 && name.StartsWith("G");
Console.WriteLine(expr);
Console.WriteLine(modifiedExpr);
O código cria uma expressão que contém uma operação AND condicional. Em seguida, ele cria uma
instância da classe AndAlsoModifier e passa a expressão ao método Modify dessa classe. A árvore de
expressão original e a modificada são geradas para mostrar a alteração.
6. Compile e execute o aplicativo.
Consulte também
Como executar árvores de expressão (C#)
Árvores de expressão (C#)
Como usar árvores de expressão para criar consultas
dinâmicasC#()
25/11/2019 • 5 minutes to read • Edit Online
No LINQ, as árvores de expressão são usadas para representar consultas estruturadas que se destinam a fontes
de dados que implementam IQueryable<T>. Por exemplo, o provedor LINQ implementa a interface
IQueryable<T> para consultar repositórios de dados relacionais. O compilador do C# compila as consultas que se
destinam a essas fontes de dados, no código que cria uma árvore de expressão em runtime. O provedor de
consultas pode percorrer a estrutura de dados da árvore de expressão e convertê-la em uma linguagem de
consulta apropriada para a fonte de dados.
As árvores de expressão também são usadas no LINQ para representar expressões lambda que são atribuídas a
variáveis do tipo Expression<TDelegate>.
Este tópico descreve como usar árvores de expressão para criar consultas LINQ dinâmicas. As consultas
dinâmicas são úteis quando as especificações de uma consulta não são conhecidas em tempo de compilação. Por
exemplo, um aplicativo pode fornecer uma interface do usuário que permite ao usuário final especificar um ou
mais predicados para filtrar os dados. Para usar a LINQ para a consulta, esse tipo de aplicativo deve usar árvores
de expressão para criar a consulta LINQ em runtime.
Exemplo
O exemplo a seguir mostra como usar árvores de expressão para construir uma consulta em uma fonte de dados
IQueryable e, em seguida, executá-la. O código compila uma árvore de expressão para representar a consulta a
seguir:
Os métodos de fábrica no namespace System.Linq.Expressions são usados para criar árvores de expressão que
representam as expressões que compõem a consulta geral. As expressões que representam as chamadas aos
métodos do operador de consulta padrão, referem-se às implementações Queryable dos métodos a seguir. A
árvore de expressão final é passada para a implementação de CreateQuery<TElement>(Expression) do provedor
da fonte de dados IQueryable para criar uma consulta executável do tipo IQueryable . Os resultados são obtidos
ao enumerar essa variável de consulta.
string[] companies = { "Consolidated Messenger", "Alpine Ski House", "Southridge Video", "City Power & Light",
"Coho Winery", "Wide World Importers", "Graphic Design Institute", "Adventure Works",
"Humongous Insurance", "Woodgrove Bank", "Margie's Travel", "Northwind Traders",
"Blue Yonder Airlines", "Trey Research", "The Phone Company",
"Wingtip Toys", "Lucerne Publishing", "Fourth Coffee" };
// Compose the expression tree that represents the parameter to the predicate.
ParameterExpression pe = Expression.Parameter(typeof(string), "company");
// ***** Where(company => (company.ToLower() == "coho winery" || company.Length > 16)) *****
// Create an expression tree that represents the expression 'company.ToLower() == "coho winery"'.
Expression left = Expression.Call(pe, typeof(string).GetMethod("ToLower", System.Type.EmptyTypes));
Expression left = Expression.Call(pe, typeof(string).GetMethod("ToLower", System.Type.EmptyTypes));
Expression right = Expression.Constant("coho winery");
Expression e1 = Expression.Equal(left, right);
// Create an expression tree that represents the expression 'company.Length > 16'.
left = Expression.Property(pe, typeof(string).GetProperty("Length"));
right = Expression.Constant(16, typeof(int));
Expression e2 = Expression.GreaterThan(left, right);
// Combine the expression trees to create an expression tree that represents the
// expression '(company.ToLower() == "coho winery" || company.Length > 16)'.
Expression predicateBody = Expression.OrElse(e1, e2);
Esse código usa um número fixo de expressões no predicado que é passado para o método Queryable.Where . No
entanto, você pode escrever um aplicativo que combina um número variável de expressões de predicado que
dependam da entrada do usuário. Você também pode variar os operadores de consulta padrão que são chamados
na consulta, dependendo da entrada do usuário.
Compilando o código
Inclua o namespace System.Linq.Expressions.
Consulte também
Árvores de expressão (C#)
Como executar árvores de expressão (C#)
Como especificar filtros predicados dinamicamente em runtime
Depurando árvores de expressão no Visual Studio
(C#)
31/10/2019 • 2 minutes to read • Edit Online
Ao depurar seus aplicativos, você pode analisar a estrutura e o conteúdo das árvores de expressão. Para obter uma
visão geral da estrutura de árvore de expressão, você pode usar a propriedade DebugView , que representa as
árvores de expressão usando uma sintaxe especial. (Observe que DebugView está disponível apenas no modo de
depuração.)
Uma vez que DebugView é uma cadeia de caracteres, você pode usar o Visualizador de Texto interno para exibi-lo
em várias linhas, selecionando Visualizador de Texto do ícone de lupa ao lado do rótulo DebugView .
Como alternativa, você pode instalar e usar um visualizador personalizado para árvores de expressão, como:
As Expressões Legíveis (licença do MIT, disponível no Visual Studio Marketplace) renderizam a árvore de
expressão com código em C#:
O Visualizador de Árvore de Expressão (licença do MIT) fornece uma exibição gráfica da árvore de
expressão, suas propriedades e os objetos relacionados:
Consulte também
Árvores de expressão (C#)
Depurando no Visual Studio
Criar visualizadores personalizados
Sintaxe do DebugView
Sintaxe de DebugView
23/10/2019 • 3 minutes to read • Edit Online
A propriedade DebugView (disponível apenas durante a depuração) fornece uma renderização de cadeia de
caracteres de árvores de expressão. A maior parte da sintaxe é bastante simples de entender; os casos especiais
são descritos nas seções a seguir.
Cada exemplo é seguido por um comentário de bloco, que contém DebugView .
ParameterExpression
System.Linq.Expressions.ParameterExpression os nomes de variáveis são exibidos com o símbolo $ no início.
Se um parâmetro não tiver um nome, será atribuído um nome gerado automaticamente, como $var1 ou $var2 .
Exemplos
ConstantExpression
Para objetos System.Linq.Expressions.ConstantExpression que representam valores inteiros, cadeias de caracteres
e null , o valor da constante é exibido.
Para tipos numéricos que têm sufixos padrão como literais de C#, o sufixo é adicionado ao valor. A tabela a seguir
mostra os sufixos associados com vários tipos numéricos.
System.UInt32 uint U
System.Int64 long L
System.UInt64 ulong UL
System.Double double D
System.Single float F
System.Decimal decimal M
Exemplos
int num = 10;
ConstantExpression expr = Expression.Constant(num);
/*
10
*/
BlockExpression
Se o tipo de um objeto System.Linq.Expressions.BlockExpression difere do tipo da última expressão no bloco, o
tipo será exibido entre colchetes angulares ( < e > ). Caso contrário, o tipo do objeto BlockExpression não é
exibido.
Exemplos
LambdaExpression
Objetos System.Linq.Expressions.LambdaExpression são exibidos junto com seus tipos delegados.
Se uma expressão lambda não tiver um nome, será atribuído um nome gerado automaticamente, como #Lambda1
ou #Lambda2 .
Exemplos
LabelExpression
Se você especificar um valor padrão para o objeto System.Linq.Expressions.LabelExpression, esse valor será
exibido antes do objeto System.Linq.Expressions.LabelTarget.
O token .Label indica o início do rótulo. O token .LabelTarget indica o local para o qual o destino deve saltar.
Se um rótulo não tiver um nome, será atribuído um nome gerado automaticamente, como #Label1 ou #Label2 .
Exemplos
Operadores verificados
Os operadores verificados são exibidos com o símbolo # na frente do operador. Por exemplo, o operador de
adição verificado é exibido como #+ .
Exemplos
Um iterador pode ser usado para percorrer coleções, como listas e matrizes.
Um método iterador ou um acessador get realiza uma iteração personalizada em uma coleção. Um método
iterador usa a instrução yield return para retornar um elemento de cada vez. Quando uma instrução
yield return for atingida, o local atual no código será lembrado. A execução será reiniciada desse local na
próxima vez que a função iteradora for chamada.
Um iterador é consumido no código cliente, usando uma instrução foreach ou usando uma consulta LINQ.
No exemplo a seguir, a primeira iteração do loop foreach faz que a execução continue no método iterador
SomeNumbers até que a primeira instrução yield return seja alcançada. Essa iteração retorna um valor de 3 e o
local atual no método iterador é mantido. Na próxima iteração do loop, a execução no método iterador
continuará de onde parou, parando novamente quando alcançar uma instrução yield return . Essa iteração
retorna um valor de 5 e o local atual no método iterador é mantido novamente. O loop terminará quando o final
do método iterador for alcançado.
O tipo de retorno de um método iterador ou acessador get pode ser IEnumerable, IEnumerable<T>,
IEnumerator ou IEnumerator<T>.
Você pode usar uma instrução yield break para terminar a iteração.
NOTE
Todos os exemplos neste tópico, exceto o exemplo Iterador Simples, incluem diretivas using para os namespaces
System.Collections e System.Collections.Generic .
Iterador simples
O exemplo a seguir contém uma única instrução yield return que está dentro de um loop for. Em Main , cada
iteração do corpo da instrução foreach cria uma chamada à função iteradora, que avança para a próxima
instrução yield return .
static void Main()
{
foreach (int number in EvenSequence(5, 18))
{
Console.Write(number.ToString() + " ");
}
// Output: 6 8 10 12 14 16 18
Console.ReadKey();
}
O exemplo a seguir cria uma classe Zoo que contém uma coleção de animais.
A instrução foreach, que faz referência à instância de classe ( theZoo ), chama implicitamente o método
GetEnumerator . As instruções foreach , que fazem referência às propriedades Birds e Mammals , usam o método
iterador nomeado AnimalsForType .
static void Main()
{
Zoo theZoo = new Zoo();
theZoo.AddMammal("Whale");
theZoo.AddMammal("Rhinoceros");
theZoo.AddBird("Penguin");
theZoo.AddBird("Warbler");
Console.ReadKey();
}
// Public methods.
public void AddMammal(string name)
{
animals.Add(new Animal { Name = name, Type = Animal.TypeEnum.Mammal });
}
// Public members.
public IEnumerable Mammals
{
get { return AnimalsForType(Animal.TypeEnum.Mammal); }
}
// Private methods.
// Private methods.
private IEnumerable AnimalsForType(Animal.TypeEnum type)
{
foreach (Animal theAnimal in animals)
{
if (theAnimal.Type == type)
{
yield return theAnimal.Name;
}
}
}
// Private class.
private class Animal
{
public enum TypeEnum { Bird, Mammal }
Console.ReadKey();
}
IEnumerator IEnumerable.GetEnumerator()
{
return GetEnumerator();
}
Informações de sintaxe
Um iterador pode ocorrer como um método ou como um acessador get . Um iterador não pode ocorrer em um
evento, um construtor de instância, um construtor estático ou um finalizador estático.
Deve haver uma conversão implícita do tipo de expressão na instrução yield return para o argumento de tipo
no IEnumerable<T> retornado pelo iterador.
Em C#, um método iterador não pode ter os parâmetros in , ref nem out .
No C#, a "yield" não é uma palavra reservada e, só terá um significado especial, quando for usada antes de uma
palavra-chave return ou break .
Implementação Técnica
Embora você escreva um iterador como um método, o compilador o traduz em uma classe aninhada que é, na
verdade, uma máquina de estado. Essa classe mantém o controle da posição do iterador enquanto o loop
foreach no código cliente continuar.
Para ver o que o compilador faz, você pode usar a ferramenta Ildasm.exe para exibir o código Microsoft
Intermediate Language que é gerado para um método iterador.
Quando você cria um iterador para uma classe ou struct, não é necessário implementar toda a interface
IEnumerator. Quando o compilador detecta o iterador, ele gera automaticamente os métodos Current ,
MoveNext e Dispose da interface IEnumerator ou IEnumerator<T>.
A cada iteração sucessiva do loop foreach (ou a chamada direta ao IEnumerator.MoveNext ), o próximo corpo de
código do iterador continua, depois da instrução yield return anterior. Em seguida, ele continuará até a
próxima instrução yield return , até que o final do corpo do iterador seja alcançado ou até que uma instrução
yield break seja encontrada.
Iteradores não dão suporte ao método IEnumerator.Reset. Para iterar novamente desde o início, você deve obter
um novo iterador. Chamar Reset no iterador retornado por um método iterador lança um
NotSupportedException.
Para obter informações adicionais, consulte a Especificação da linguagem C#.
Uso de iteradores
Os iteradores permitem que você mantenha a simplicidade de um loop foreach quando for necessário usar um
código complexo para preencher uma sequência de lista. Isso pode ser útil quando você quiser fazer o seguinte:
Modificar a sequência de lista após a primeira iteração de loop foreach .
Evitar o carregamento completo de uma grande lista antes da primeira iteração de um loop foreach . Um
exemplo é uma busca paginada para carregar um lote de linhas da tabela. Outro exemplo é o método
EnumerateFiles, que implementa os iteradores dentro do .NET Framework.
Encapsular a criação da lista no iterador. No método iterador, você pode criar a lista e, em seguida, gerar
cada resultado em um loop.
Consulte também
System.Collections.Generic
IEnumerable<T>
foreach, in
yield
Usando foreach com matrizes
Genéricos
LINQ (Consulta Integrada à Linguagem)
04/11/2019 • 7 minutes to read • Edit Online
O LINQ (consulta integrada à linguagem) é o nome de um conjunto de tecnologias com base na integração
de recursos de consulta diretamente na linguagem C#. Tradicionalmente, consultas em dados são expressas
como cadeias de caracteres simples sem verificação de tipo no tempo de compilação ou no suporte a
IntelliSense. Além disso, você terá de aprender uma linguagem de consulta diferente para cada tipo de fonte
de dados: bancos de dados SQL, documentos XML, vários serviços Web e etc. Com o LINQ, uma consulta é
um constructo de linguagem de primeira classe, como classes, métodos, eventos. Você escreve consultas em
coleções fortemente tipadas de objetos usando palavras-chave da linguagem e operadores familiares. A
família de tecnologias LINQ fornece uma experiência de consulta consistente para objetos (LINQ to Objects),
bancos de dados relacionais (LINQ to SQL ) e XML (LINQ to XML ).
Para um desenvolvedor que escreve consultas, a parte mais visível "integrada à linguagem" do LINQ é a
expressão de consulta. As expressões de consulta são uma sintaxe declarativa de consulta. Usando a sintaxe
de consulta, você pode executar operações de filtragem, ordenação e agrupamento em fontes de dados com o
mínimo de código. Você pode usar os mesmos padrões de expressão de consulta básica para consultar e
transformar dados em bancos de dados SQL, conjuntos de dados do ADO.NET, documentos XML e fluxos e
coleções .NET.
É possível escrever consultas do LINQ em C# para bancos de dados do SQL Server, documentos XML,
conjuntos de dados ADO.NET e qualquer coleção de objetos que dá suporte a IEnumerable ou à interface
genérica IEnumerable<T>. O suporte ao LINQ também é fornecido por terceiros para muitos serviços Web e
outras implementações de banco de dados.
O exemplo a seguir mostra a operação de consulta completa. A operação completa inclui a criação de uma
fonte de dados, definição da expressão de consulta e execução da consulta em uma instrução foreach .
class LINQQueryExpressions
{
static void Main()
{
A seguinte ilustração do Visual Studio mostra uma consulta LINQ parcialmente concluída em um banco de
dados do SQL Server no C# e no Visual Basic, com verificação de tipo completa e suporte ao IntelliSense:
Visão geral da expressão de consulta
Expressões de consulta podem ser usadas para consultar e transformar dados de qualquer fonte de dados
habilitada para LINQ. Por exemplo, uma única consulta pode recuperar dados de um Banco de Dados SQL
e produzir um fluxo XML como saída.
As expressões de consulta são fáceis de entender porque elas usam muitos constructos de linguagem C#
familiares.
As variáveis em uma expressão de consulta são fortemente tipadas, embora em muitos casos você não
precise fornecer o tipo explicitamente, pois o compilador pode inferir nele. Para obter mais informações,
consulte Relacionamentos de tipo em operações de consulta LINQ.
Uma consulta não é executada até que você itere sobre a variável de consulta, por exemplo, em uma
instrução foreach . Para obter mais informações, consulte Introdução a consultas LINQ.
No tempo de compilação, as expressões de consulta são convertidas em chamadas de método do
operador de consulta padrão de acordo com as regras definidas na especificação do C#. Qualquer consulta
que pode ser expressa usando sintaxe de consulta também pode ser expressa usando sintaxe de método.
No entanto, na maioria dos casos, a sintaxe de consulta é mais legível e concisa. Para obter mais
informações, consulte Especificação da linguagem C# e Visão geral de operadores de consulta padrão.
Como uma regra ao escrever consultas LINQ, recomendamos que você use a sintaxe de consulta sempre
que possível e a sintaxe de método sempre que necessário. Não há semântica ou diferença de
desempenho entre as duas formas. As expressões de consulta são geralmente mais legíveis do que as
expressões equivalentes escritas na sintaxe de método.
Algumas operações de consulta, como Count ou Max, não apresentam cláusulas de expressão de consulta
equivalentes e, portanto, devem ser expressas como chamadas de método. A sintaxe de método pode ser
combinada com a sintaxe de consulta de várias maneiras. Para obter mais informações, consulte Sintaxe de
consulta e sintaxe de método em LINQ.
As expressões de consulta podem ser compiladas para árvores de expressão ou delegados, dependendo
do tipo ao qual a consulta é aplicada. As consultas IEnumerable<T> são compiladas para representantes.
As consultas IQueryable e IQueryable<T> são compiladas para árvores de expressão. Para obter mais
informações, consulte Árvores de expressão.
Próximas etapas
Para obter mais detalhes sobre o LINQ, comece se familiarizando com alguns conceitos básicos em Noções
básicas sobre expressões de consulta, e, em seguida, leia a documentação para a tecnologia LINQ na qual
você está interessado:
Documentos XML: LINQ to XML
ADO.NET Entity Framework: LINQ to Entities
Coleções do .NET, arquivos, cadeias de caracteres, etc.: LINQ to Objects
Para saber mais sobre o LINQ, consulte LINQ em C#.
Para começar a trabalhar com o LINQ em C#, consulte o tutorial Trabalhando com LINQ.
Introdução a consultas LINQ (C#)
04/11/2019 • 11 minutes to read • Edit Online
Uma consulta é uma expressão que recupera dados de uma fonte de dados. As consultas normalmente são
expressas em uma linguagem de consulta especializada. Diferentes linguagens foram desenvolvidas ao longo
do tempo para os diversos tipos de fontes de dados, por exemplo, SQL para bancos de dados relacionais e o
XQuery para XML. Portanto, os desenvolvedores precisaram aprender uma nova linguagem de consulta para
cada tipo de fonte de dados ou formato de dados que eles tinham que oferecer suporte. O LINQ simplifica essa
situação ao oferecer um modelo consistente para trabalhar com os dados entre vários tipos de fontes e
formatos de dados. Em uma consulta LINQ, você está sempre trabalhando com objetos. Você usa os mesmos
padrões básicos de codificação para consultar e transformar dados em documentos XML, bancos de dados
SQL, conjuntos de dados do ADO.NET coleções do .NET e qualquer outro formato para o qual um provedor
LINQ estiver disponível.
class IntroToLINQ
{
static void Main()
{
// The Three Parts of a LINQ Query:
// 1. Data source.
int[] numbers = new int[7] { 0, 1, 2, 3, 4, 5, 6 };
// 2. Query creation.
// numQuery is an IEnumerable<int>
var numQuery =
from num in numbers
where (num % 2) == 0
select num;
// 3. Query execution.
foreach (int num in numQuery)
{
Console.Write("{0,1} ", num);
}
}
}
A ilustração a seguir mostra a operação de consulta completa. No LINQ, a execução da consulta é diferente da
própria consulta. Em outras palavras, você não recupera dados apenas criando uma variável de consulta.
A Fonte de Dados
No exemplo anterior, como a fonte de dados é uma matriz, ela dá suporte à interface genérica IEnumerable<T>
de forma implícita. Isso significa que ela pode ser consultada com o LINQ. Uma consulta é executada em uma
instrução foreach , e foreach requer IEnumerable ou IEnumerable<T>. Tipos que dão suporte a
IEnumerable<T> ou uma interface derivada, como a genérica IQueryable<T>, são chamados tipos passíveis de
consulta.
Um tipo passível de consulta não exige modificação ou tratamento especial para servir como uma fonte de
dados do LINQ. Se os dados de origem ainda não estiverem na memória como um tipo passível de consulta, o
provedor do LINQ deverá representá-los como tal. Por exemplo, LINQ to XML carrega um documento XML em
um tipo XElement passível de consulta:
Com o LINQ to SQL, primeiro você cria um mapeamento relacional de objeto em tempo de design,
manualmente ou usando as Ferramentas LINQ to SQL no Visual Studio. Você escreve suas consultas aos
objetos e o LINQ to SQL manipula a comunicação com o banco de dados em tempo de execução. No exemplo
a seguir, Customers representa uma tabela específica no banco de dados, e o tipo do resultado da consulta,
IQueryable<T>, deriva de IEnumerable<T>.
Para obter mais informações sobre como criar tipos específicos de fontes de dados, consulte a documentação
para os diversos provedores do LINQ. No entanto, a regra básica é muito simples: uma fonte de dados LINQ é
qualquer objeto que dá suporte à interface genérica IEnumerable<T> ou uma interface que herda dela.
NOTE
Tipos, como ArrayList, que dão suporte à interface IEnumerable não genérica também podem ser usados como uma fonte
de dados LINQ. Para obter mais informações, consulte Como consultar um ArrayList com LINQ (C#).
A consulta
A consulta especifica quais informações devem ser recuperadas da fonte (ou fontes) de dados. Opcionalmente,
uma consulta também especifica como essas informações devem ser classificadas, agrupadas e moldadas antes
de serem retornadas. Uma consulta é armazenada em uma variável de consulta e é inicializada com uma
expressão de consulta. Para tornar mais fácil escrever consultas, o C# introduziu uma nova sintaxe de consulta.
A consulta no exemplo anterior retorna todos os números pares da matriz de inteiros. A expressão de consulta
contém três cláusulas: from , where e select . (Se você estiver familiarizado com o SQL, perceberá que a
ordenação das cláusulas é revertida da ordem em SQL.) A cláusula from especifica a fonte de dados, a cláusula
where aplica o filtro e a cláusula select especifica o tipo dos elementos retornados. Essas e outras cláusulas de
consulta são discutidas detalhadamente na seção Expressões de consulta LINQ. Por enquanto, o ponto
importante é que, no LINQ, a variável de consulta não faz nada e não retorna nenhum dado. Ele apenas
armazena as informações necessárias para produzir os resultados quando a consulta for executada em um
momento posterior. Para obter mais informações sobre como as consultas são construídas nos bastidores,
consulte Visão geral de operadores de consulta padrão (C#).
NOTE
As consultas também podem ser expressas usando a sintaxe de método. Para obter mais informações, consulte Sintaxe de
consulta e sintaxe de método em LINQ.
Execução da Consulta
Execução Adiada
Conforme mencionado anteriormente, a variável de consulta armazena somente os comandos da consulta. A
execução real da consulta é adiada até que você itere sobre a variável de consulta em uma instrução foreach .
Esse conceito é conhecido como execução adiada e é demonstrado no exemplo a seguir:
// Query execution.
foreach (int num in numQuery)
{
Console.Write("{0,1} ", num);
}
A instrução foreach também é o local em que os resultados da consulta são recuperados. Por exemplo, na
consulta anterior, a variável de iteração num armazena cada valor (um de cada vez) na sequência retornada.
Como a própria variável de consulta nunca armazena os resultados da consulta, você poderá executá-la quantas
vezes desejar. Por exemplo, você pode ter um banco de dados que está sendo atualizado continuamente por um
aplicativo separado. Em seu aplicativo, você poderia criar uma consulta que recupera os dados mais recentes e
poderia executá-la repetidamente em algum intervalo para recuperar resultados diferentes a cada vez.
Forçando Execução Imediata
As consultas que realizam funções de agregação em um intervalo de elementos de origem devem primeiro
iterar sobre esses elementos. Exemplos dessas consultas são Count , Max , Average e First . Essas consultas
são executadas sem uma instrução foreach explícita porque a consulta em si deve usar foreach para retornar
um resultado. Observe também que esses tipos de consultas retornam um valor único e não uma coleção
IEnumerable . A consulta a seguir retorna uma contagem de números pares na matriz de origem:
var evenNumQuery =
from num in numbers
where (num % 2) == 0
select num;
Para forçar a execução imediata de qualquer consulta e armazenar seus resultados em cache, você pode chamar
os métodos ToList ou ToArray.
List<int> numQuery2 =
(from num in numbers
where (num % 2) == 0
select num).ToList();
// or like this:
// numQuery3 is still an int[]
var numQuery3 =
(from num in numbers
where (num % 2) == 0
select num).ToArray();
Você também pode forçar a execução colocando o loop foreach imediatamente após a expressão de consulta.
No entanto, ao chamar ToList ou ToArray , você também armazena em cache todos os dados em um único
objeto de coleção.
Consulte também
Introdução a LINQ em C#
Passo a passo: escrevendo consultas em C#
Expressões de consulta LINQ
foreach, in
Palavras-chave de Consulta (LINQ )
LINQ e tipos genéricos (C#)
23/10/2019 • 3 minutes to read • Edit Online
As consultas do LINQ são baseadas em tipos genéricos, que foram introduzidos na versão 2.0 do .NET
Framework. Não é necessário um conhecimento profundo sobre os genéricos antes de começar a escrever
consultas. No entanto, convém entender dois conceitos básicos:
1. Quando você cria uma instância de uma classe de coleção genérica, como List<T>, substitua o "T" pelo tipo
dos objetos que a lista bloqueia. Por exemplo, uma lista de cadeias de caracteres é expressa como
List<string> e uma lista de objetos Customer é expressa como List<Customer> . Uma lista genérica é
fortemente tipada e oferece muitos benefícios em coleções que armazenam seus elementos como Object.
Se tentar adicionar um Customer em uma List<string> , você obterá um erro em tempo de compilação. É
fácil usar coleções genéricas, porque você não precisa realizar a conversão de tipo em tempo de execução.
2. A IEnumerable<T> é a interface que permite que as classes de coleção genérica sejam enumeradas usando
a instrução foreach . Classes de coleção genéricas dão suporte a IEnumerable<T> do mesmo modo que
classes de coleção não genéricas, tais como ArrayList, dão suporte a IEnumerable.
Para obter mais informações sobre os genéricos, consulte Genéricos.
IEnumerable<Customer> customerQuery =
from cust in customers
where cust.City == "London"
select cust;
Para obter mais informações, consulte Relacionamentos de tipo em operações de consulta LINQ.
A palavra-chave var é útil quando o tipo da variável for óbvio ou quando não é tão importante especificar
explicitamente os tipos genéricos aninhados, como aqueles que são produzidos por consultas de grupo. É
recomendável que você note que o código poderá se tornar mais difícil de ser lido por outras pessoas, caso você
use a var . Para obter mais informações, consulte Variáveis locais de tipo implícito.
Consulte também
Genéricos
Operações de consulta LINQ básica (C#)
04/11/2019 • 9 minutes to read • Edit Online
Este tópico fornece uma breve introdução às expressões de consulta LINQ e a alguns dos tipos típicos de
operações que podem ser executadas em uma consulta. Informações mais detalhadas estão nos tópicos a seguir:
Expressões de consulta LINQ
Visão geral de operadores de consulta padrão (C#)
Passo a passo: escrevendo consultas em C#
NOTE
Se já estiver familiarizado com uma linguagem de consulta como SQL ou XQuery, você poderá ignorar a maior parte deste
tópico. Leia sobre a "cláusula from " na próxima seção para saber mais sobre a ordem das cláusulas em expressões de
consulta LINQ.
//queryAllCustomers is an IEnumerable<Customer>
var queryAllCustomers = from cust in customers
select cust;
A variável de intervalo é como a variável de iteração em um loop foreach , mas nenhuma iteração real ocorre em
uma expressão de consulta. Quando a consulta é executada, a variável de intervalo servirá como uma referência
para cada elemento sucessivo em customers . Uma vez que o compilador pode inferir o tipo de cust , você não
precisa especificá-lo explicitamente. Variáveis de intervalo adicionais podem ser introduzidas por uma cláusula
let . Para obter mais informações, consulte Cláusula let.
NOTE
Para fontes de dados não genéricas, como ArrayList, a variável de intervalo deve ser tipada explicitamente. Para obter mais
informações, consulte Como consultar um ArrayList com LINQ (C#) e Cláusula from.
Filtragem
Provavelmente, a operação de consulta mais comum é aplicar um filtro no formulário de uma expressão booliana.
O filtro faz com que a consulta retorne apenas os elementos para os quais a expressão é verdadeira. O resultado é
produzido usando a cláusula where . O filtro em vigor especifica os elementos a serem excluídos da sequência de
origem. No exemplo a seguir, somente os customers que têm um endereço em Londres são retornados.
Ordenando
Muitas vezes é conveniente classificar os dados retornados. A cláusula orderby fará com que os elementos na
sequência retornada sejam classificados de acordo com o comparador padrão para o tipo que está sendo
classificado. Por exemplo, a consulta a seguir pode ser estendida para classificar os resultados com base na
propriedade Name . Como Name é uma cadeia de caracteres, o comparador padrão executa uma classificação em
ordem alfabética de A a Z.
var queryLondonCustomers3 =
from cust in customers
where cust.City == "London"
orderby cust.Name ascending
select cust;
Agrupamento
A cláusula group permite agrupar seus resultados com base em uma chave que você especificar. Por exemplo,
você pode especificar que os resultados sejam agrupados segundo o City , de modo que todos os clientes de
Londres ou Paris fiquem em grupos individuais. Nesse caso, cust.City é a chave.
Quando você terminar uma consulta com um cláusula group , seus resultados assumirão a forma de uma lista de
listas. Cada elemento na lista é um objeto que tem um membro Key e uma lista dos elementos que estão
agrupados sob essa chave. Quando itera em uma consulta que produz uma sequência de grupos, você deve usar
um loop foreach aninhado. O loop externo itera em cada grupo e o loop interno itera nos membros de cada
grupo.
Se precisar consultar os resultados de uma operação de grupo, você poderá usar a palavra-chave into para criar
um identificador que pode ser consultado ainda mais. A consulta a seguir retorna apenas os grupos que contêm
mais de dois clientes:
Ingressando
Operações de junção criam associações entre sequências que não são modeladas explicitamente nas fontes de
dados. Por exemplo, você pode executar uma junção para localizar todos os clientes e distribuidores que têm o
mesmo local. Em LINQ, a cláusula join sempre funciona com coleções de objetos em vez de tabelas de banco de
dados diretamente.
var innerJoinQuery =
from cust in customers
join dist in distributors on cust.City equals dist.City
select new { CustomerName = cust.Name, DistributorName = dist.Name };
Em LINQ, você não precisa usar join com a mesma frequência que o faz no SQL, porque as chaves estrangeiras
em LINQ são representados no modelo do objeto como propriedades que mantêm uma coleção de itens. Por
exemplo, um objeto Customer que contém uma coleção de objetos Order . Em vez de executar uma junção, você
pode acessar os pedidos usando notação de ponto:
Selecionando (Projeções)
A cláusula select produz os resultados da consulta e especifica a "forma" ou o tipo de cada elemento retornado.
Por exemplo, você pode especificar se os resultados consistirão em objetos Customer completos, apenas um
membro, um subconjunto de membros ou algum tipo de resultado completamente diferente com base em um
cálculo ou na criação de um novo objeto. Quando a cláusula select produz algo diferente de uma cópia do
elemento de origem, a operação é chamada de projeção. O uso de projeções para transformar dados é uma
funcionalidade poderosa das expressões de consulta de LINQ. Para obter mais informações, consulte
Transformações de dados com LINQ (C#) e Cláusula select.
Consulte também
Expressões de consulta LINQ
Passo a passo: escrevendo consultas em C#
Palavras-chave de Consulta (LINQ )
Tipos Anônimos
Transformações de dados com LINQ (C#)
04/11/2019 • 8 minutes to read • Edit Online
LINQ (Consulta Integrada à Linguagem) não se trata apenas de recuperação de dados. Também é uma
ferramenta poderosa para transformação de dados. Ao usar uma consulta LINQ, você pode usar uma sequência
de origem como entrada e modificá-la de várias maneiras para criar uma nova sequência de saída. Você pode
modificar a própria sequência sem modificar os respectivos elementos, classificando-os e agrupando-os. Mas
talvez o recurso mais poderoso das consultas LINQ é a capacidade de criar novos tipos. Isso é feito na cláusula
select. Por exemplo, é possível executar as seguintes tarefas:
Mesclar várias sequências de entrada em uma única sequência de saída que tenha um novo tipo.
Criar sequências de saída cujos elementos consistem em apenas uma ou várias propriedades de cada
elemento da sequência de origem.
Criar sequências de saída cujos elementos consistem nos resultados das operações realizadas nos dados
de origem.
Criar sequências de saída em um formato diferente. Por exemplo, você pode transformar dados de linhas
do SQL ou de arquivos de texto em XML.
Esses são apenas alguns exemplos. É claro que essas transformações podem ser combinadas de diversas
maneiras na mesma consulta. Além disso, a sequência de saída de uma consulta pode ser usada como a
sequência de entrada de uma nova consulta.
class Student
{
public string First { get; set; }
public string Last {get; set;}
public int ID { get; set; }
public string Street { get; set; }
public string City { get; set; }
public List<int> Scores;
}
class Teacher
{
public string First { get; set; }
public string Last { get; set; }
public int ID { get; set; }
public string City { get; set; }
}
2. Para criar elementos que contenham mais de uma propriedade do elemento de origem, você pode usar um
inicializador de objeto com um objeto nomeado ou um tipo anônimo. O exemplo a seguir mostra o uso de
um tipo anônimo para encapsular duas propriedades de cada elemento Customer :
Para obter mais informações, consulte Inicializadores de coleção e de objeto e Tipos anônimos.
class XMLTransform
{
static void Main()
{
// Create the data source by using a collection initializer.
// The Student class was defined previously in this topic.
List<Student> students = new List<Student>()
{
new Student {First="Svetlana", Last="Omelchenko", ID=111, Scores = new List<int>{97, 92, 81,
60}},
new Student {First="Claire", Last="O’Donnell", ID=112, Scores = new List<int>{75, 84, 91, 39}},
new Student {First="Sven", Last="Mortensen", ID=113, Scores = new List<int>{88, 94, 65, 91}},
};
Para obter mais informações, consulte Criando árvores XML em C# (LINQ to XML ) .
NOTE
Não há suporte para chamar métodos em expressões de consulta se a consulta será movida para outro domínio. Por
exemplo, você não pode chamar um método comum de C# no LINQ to SQL porque o SQL Server não tem contexto para
ele. No entanto, você pode mapear procedimentos armazenados para os métodos e chamá-los. Para obter mais
informações, consulte Procedimentos armazenados.
class FormatQuery
{
static void Main()
{
// Data source.
double[] radii = { 1, 2, 3 };
// Query.
IEnumerable<string> query =
from rad in radii
select $"Area = {rad * rad * Math.PI:F2}";
// Query execution.
foreach (string s in query)
Console.WriteLine(s);
Para escrever consultas com eficiência, você precisa entender como os tipos de variáveis em uma operação de
consulta completa se relacionam entre si. Se compreender esses relacionamentos, você compreenderá com
maior facilidade os exemplos de LINQ e exemplos de código na documentação. Além disso, você compreenderá
o que ocorre nos bastidores quando variáveis são tipadas de forma implícita usando var .
Operações de consulta de LINQ são fortemente tipadas na fonte de dados, na própria consulta e na execução
da consulta. O tipo das variáveis na consulta deve ser compatível com o tipo dos elementos na fonte de dados e
com o tipo da variável de iteração na instrução foreach . Essa tipagem forte garante que erros de tipo sejam
capturados em tempo de compilação, quando podem ser corrigidos antes que os usuários os encontrem.
Para demonstrar essas relações de tipo, a maioria dos exemplos a seguir usam tipagem explícita para todas as
variáveis. O último exemplo mostra como os mesmos princípios se aplicam mesmo quando você usa tipagem
implícita usando var.
3. A variável de consulta é iterada na instrução foreach . Como a variável de consulta é uma sequência de
cadeias de caracteres, a variável de iteração também é uma cadeia de caracteres.
A maioria das consultas na documentação introdutória da LINQ (Consulta Integrada à Linguagem) é escrita
usando a sintaxe de consulta declarativa da LINQ. No entanto, a sintaxe de consulta deve ser convertida em
chamadas de método para o CLR (Common Language Runtime) do .NET quando o código for compilado.
Essas chamadas de método invocam os operadores de consulta padrão, que têm nomes como Where , Select ,
GroupBy , Join , Max e Average . Você pode chamá-los diretamente usando a sintaxe de método em vez da
sintaxe de consulta.
A sintaxe de consulta e a sintaxe de método são semanticamente idênticas, mas muitas pessoas acham a
sintaxe de consulta mais simples e fácil de ler. Algumas consultas devem ser expressadas como chamadas de
método. Por exemplo, você deve usar uma chamada de método para expressar uma consulta que recupera o
número de elementos que correspondem a uma condição especificada. Você também deve usar uma chamada
de método para uma consulta que recupera o elemento que tem o valor máximo em uma sequência de origem.
A documentação de referência para os operadores de consulta padrão no namespace System.Linq geralmente
usa a sintaxe de método. Portanto, mesmo quando começar a escrever consultas LINQ, é útil estar
familiarizado com como usar a sintaxe de método nas consultas e nas próprias expressões de consulta.
//Query syntax:
IEnumerable<int> numQuery1 =
from num in numbers
where num % 2 == 0
orderby num
select num;
//Method syntax:
IEnumerable<int> numQuery2 = numbers.Where(num => num % 2 == 0).OrderBy(n => n);
A saída dos dois exemplos é idêntica. Você pode ver que o tipo da variável de consulta é o mesmo em ambas as
formas: IEnumerable<T>.
Para entender a consulta baseada em método, vamos examiná-la melhor. No lado direito da expressão, observe
que a cláusula where agora é expressa como um método de instância no objeto numbers , que, como você deve
se lembrar, tem um tipo de IEnumerable<int> . Se você estiver familiarizado com a interface IEnumerable<T>
genérica, você saberá que ela não tem um método Where . No entanto, se você invocar a lista de conclusão do
IntelliSense no IDE do Visual Studio, verá não apenas um método Where , mas muitos outros métodos como
Select , SelectMany , Join e Orderby . Esses são todos os operadores de consulta padrão.
Embora pareça como se IEnumerable<T> tivesse sido redefinido para incluir esses métodos adicionais, na
verdade esse não é o caso. Os operadores de consulta padrão são implementados como um novo tipo de
método chamado métodos de extensão. Métodos de extensão "estendem" um tipo existente, eles podem ser
chamados como se fossem métodos de instância no tipo. Os operadores de consulta padrão estendem
IEnumerable<T> e que é por esse motivo que você pode escrever numbers.Where(...) .
Para começar a usar LINQ, tudo o que você realmente precisa saber sobre os métodos de extensão é como
colocá-los no escopo em seu aplicativo usando as diretivas using corretas. Do ponto de vista do aplicativo, um
método de extensão e um método de instância normal são iguais.
Para obter mais informações sobre os métodos de extensão, consulte Métodos de extensão. Para obter mais
informações sobre os operadores de consulta padrão, consulte Visão geral de operadores de consulta padrão
(C#). Alguns provedores de LINQ, como LINQ to SQL e LINQ to XML, implementam seus próprios
operadores de consulta padrão e os métodos de extensão adicionais para outros tipos além de
IEnumerable<T>.
Expressões lambda
No exemplo anterior, observe que a expressão condicional ( num % 2 == 0 ) é passada como um argumento em
linha para o método Where : Where(num => num % 2 == 0). Essa expressão embutida é chamada de expressão
lambda. É uma maneira conveniente de escrever um código que de outra forma precisaria ser escrito de forma
mais complicada como um método anônimo, um delegado genérico ou uma árvore de expressão. No C# => é
o operador lambda, que é lido como "vai para". O num à esquerda do operador é a variável de entrada que
corresponde ao num na expressão de consulta. O compilador pode inferir o tipo de num porque ele sabe que
numbers é um tipo IEnumerable<T> genérico. O corpo do lambda é exatamente igual à expressão na sintaxe
de consulta ou em qualquer outra expressão ou instrução C#, ele pode incluir chamadas de método e outra
lógica complexa. O "valor retornado" é apenas o resultado da expressão.
Para começar a usar LINQ, você não precisa usar lambdas extensivamente. No entanto, determinadas
consultas só podem ser expressadas em sintaxe de método e algumas delas requerem expressões lambda.
Após você se familiarizar mais com lambdas, verá que eles são uma ferramenta poderosa e flexível na sua caixa
de ferramentas LINQ. Para obter mais informações, consulte Expressões Lambda.
A seção a seguir apresenta os novos constructos de linguagem introduzidos no C# 3.0. Embora esses novos
recursos tenham algum grau de utilização com consultas LINQ, eles não estão limitados a LINQ e podem ser
usados em qualquer contexto em que sejam úteis.
Expressões de consulta
As expressões de consulta usam uma sintaxe declarativa semelhante ao SQL ou XQuery para consultar em
coleções IEnumerable. Em tempo de compilação, a sintaxe de consulta é convertida em chamadas de método a
uma implementação da LINQ dos métodos de extensão do operador de consulta padrão do provedor. Os
aplicativos controlam os operadores de consulta padrão que estão no escopo, especificando o namespace
apropriado com uma diretiva using . A expressão de consulta a seguir pega uma matriz de cadeias de caracteres,
agrupa-os de acordo com o primeiro caractere da cadeia de caracteres e ordena os grupos.
var number = 5;
var name = "Virginia";
var query = from str in stringArray
where str[0] == 'm'
select str;
As variáveis declaradas como var são tão fortemente tipadas quanto as variáveis cujo tipo você especifica
explicitamente. O uso de var possibilita a criação de tipos anônimos, mas ele pode ser usado para quaisquer
variáveis locais. As matrizes também podem ser declaradas com tipagem implícita.
Para obter mais informações, consulte Variáveis locais de tipo implícito.
Continuando com a nossa classe Customer , suponha que haja uma fonte de dados chamada IncomingOrders e
que, para cada ordem com um grande OrderSize , desejamos criar um novo Customer com base fora dessa
ordem. Uma consulta LINQ pode ser executada nessa fonte de dados e usar a inicialização do objeto para
preencher uma coleção:
A fonte de dados pode ter mais propriedades escondidas do que a classe Customer , como OrderSize , mas com a
inicialização do objeto, os dados retornados da consulta são moldados no tipo de dados desejado; escolhemos os
dados que são relevantes para nossa classe. Consequentemente, agora temos um IEnumerable preenchido com os
novos Customer s que queríamos. O trecho acima também pode ser escrito na sintaxe de método do LINQ:
var newLargeOrderCustomers = IncomingOrders.Where(x => x.OrderSize > 5).Select(y => new Customer { Name =
y.Name, Phone = y.Phone });
Tipos anônimos
Um tipo anônimo é construído pelo compilador e o nome do tipo só fica disponível para o compilador. Os tipos
anônimos fornecem uma maneira conveniente de agrupar um conjunto de propriedades temporariamente em um
resultado de consulta, sem a necessidade de definir um tipo nomeado separado. Os tipos anônimos são
inicializados com uma nova expressão e um inicializador de objeto, como mostrado aqui:
Métodos de extensão
Um método de extensão é um método estático que pode ser associado a um tipo, para que ele possa ser chamado
como se fosse um método de instância no tipo. Esse recurso permite que você, na verdade, "adicione" novos
métodos a tipos existentes sem realmente modificá-los. Os operadores de consulta padrão são um conjunto de
métodos de extensão que fornecem a funcionalidade de consulta da LINQ para qualquer tipo que implementa a
IEnumerable<T>.
Para obter mais informações, consulte Métodos de extensão.
Expressões lambda
Uma expressão lambda é uma função embutida que usa o operador => para separar os parâmetros de entrada do
corpo da função e podem ser convertidos em um delegado ou uma árvore de expressão, em tempo de
compilação. Na programação LINQ, você encontrará as expressões lambda ao fazer chamadas de método diretas
aos operadores de consulta padrão.
Para obter mais informações, consulte:
Funções Anônimas
Expressões Lambda
Árvores de expressão (C#)
Consulte também
LINQ (consulta integrada à linguagem) (C#)
Instruções passo a passo: escrevendo consultas em
C# (LINQ)
04/11/2019 • 17 minutes to read • Edit Online
Essas instruções passo a passo demonstram os recursos de linguagem C# que são usados para gravar expressões
de consulta LINQ.
Criar um Projeto C#
NOTE
As instruções a seguir são para o Visual Studio. Se você estiver usando um ambiente de desenvolvimento diferente, crie um
projeto de console com uma referência a System.Core.dll e uma diretiva using para o namespace System.Linq.
Criar a Consulta
Para criar uma consulta simples
No método Main do aplicativo, crie uma consulta simples que, quando for executada, produzirá uma lista
de todos os alunos cuja pontuação no primeiro teste foi superior a 90. Observe que como o objeto
Student todo está selecionado, o tipo da consulta é IEnumerable<Student> . Embora o código também
possa usar a tipagem implícita usando a palavra-chave var, a tipagem explícita é usada para ilustrar
claramente os resultados. (Para obter mais informações sobre var , consulte Variáveis locais de tipo
implícito.)
Observe também que a variável de intervalo da consulta, student , também funciona como uma referência
para cada Student na fonte, fornecendo acesso ao membro para cada objeto.
Executar a Consulta
Para executar a consulta
1. Agora escreva o loop foreach que fará com que a consulta seja executada. Observe o seguinte sobre o
código:
Cada elemento na sequência retornada é acessado pela variável de iteração no loop foreach .
O tipo dessa variável é Student e o tipo da variável de consulta é compatível, IEnumerable<Student> .
2. Após você ter adicionado esse código, compile e execute o aplicativo para ver os resultados na janela
Console.
// Output:
// Omelchenko, Svetlana
// Garcia, Cesar
// Fakhouri, Fadi
// Feng, Hanying
// Garcia, Hugo
// Adams, Terry
// Zabokritski, Eugene
// Tucker, Michael
Modificar a Consulta
Para ordenar os resultados
1. Será mais fácil verificar os resultados se eles estiverem em algum tipo de ordem. Você pode ordenar a
sequência retornada por qualquer campo acessível nos elementos de origem. Por exemplo, a cláusula
orderby a seguir ordena os resultados em ordem alfabética de A a Z de acordo com o sobrenome de cada
aluno. Adicione a cláusula orderby a seguir à consulta, logo após a instrução where e antes da instrução
select :
2. Agora, altere a cláusula orderby para que ela ordene os resultados em ordem inversa de acordo com a
pontuação no primeiro teste, da pontuação mais alta para a mais baixa.
3. Altere a cadeia de caracteres de formato WriteLine para que você possa ver as pontuações:
2. Observe que o tipo da consulta agora mudou. Ele produz uma sequência de grupos que têm um tipo char
como uma chave e uma sequência de objetos Student . Como o tipo de consulta foi alterado, o código a
seguir altera o loop de execução foreach também:
// Output:
// O
// Omelchenko, Svetlana
// O'Donnell, Claire
// M
// Mortensen, Sven
// G
// Garcia, Cesar
// Garcia, Debra
// Garcia, Hugo
// F
// Fakhouri, Fadi
// Feng, Hanying
// T
// Tucker, Lance
// Tucker, Michael
// A
// Adams, Terry
// Z
// Zabokritski, Eugene
// Output:
// O
// Omelchenko, Svetlana
// O'Donnell, Claire
// M
// Mortensen, Sven
// G
// Garcia, Cesar
// Garcia, Debra
// Garcia, Hugo
// F
// Fakhouri, Fadi
// Feng, Hanying
// T
// Tucker, Lance
// Tucker, Michael
// A
// Adams, Terry
// Z
// Zabokritski, Eugene
Para obter mais informações sobre var, consulte Variáveis locais de tipo implícito.
Para ordenar os grupos pelo valor da chave
1. Ao executar a consulta anterior, observe que os grupos não estão em ordem alfabética. Para alterar isso,
você deve fornecer uma cláusula orderby após a cláusula group . Mas, para usar uma cláusula orderby ,
primeiro é necessário um identificador que serve como uma referência para os grupos criados pela
cláusula group . Forneça o identificador usando a palavra-chave into , da seguinte maneira:
var studentQuery4 =
from student in students
group student by student.Last[0] into studentGroup
orderby studentGroup.Key
select studentGroup;
// Output:
//A
// Adams, Terry
//F
// Fakhouri, Fadi
// Feng, Hanying
//G
// Garcia, Cesar
// Garcia, Debra
// Garcia, Hugo
//M
// Mortensen, Sven
//O
// Omelchenko, Svetlana
// O'Donnell, Claire
//T
// Tucker, Lance
// Tucker, Michael
//Z
// Zabokritski, Eugene
Quando você executa essa consulta, você verá que os grupos agora estão classificados em ordem
alfabética.
Para introduzir um identificador usando let
1. Você pode usar a palavra-chave let para introduzir um identificador para qualquer resultado da
expressão na expressão de consulta. Esse identificador pode ser uma conveniência, como no exemplo a
seguir ou ele pode melhorar o desempenho armazenando os resultados de uma expressão para que ele
não precise ser calculado várias vezes.
// studentQuery5 is an IEnumerable<string>
// This query returns those students whose
// first test score was higher than their
// average score.
var studentQuery5 =
from student in students
let totalScore = student.Scores[0] + student.Scores[1] +
student.Scores[2] + student.Scores[3]
where totalScore / 4 < student.Scores[0]
select student.Last + " " + student.First;
// Output:
// Omelchenko Svetlana
// O'Donnell Claire
// Mortensen Sven
// Garcia Cesar
// Fakhouri Fadi
// Feng Hanying
// Garcia Hugo
// Adams Terry
// Zabokritski Eugene
// Tucker Michael
var studentQuery6 =
from student in students
let totalScore = student.Scores[0] + student.Scores[1] +
student.Scores[2] + student.Scores[3]
select totalScore;
// Output:
// Class average score = 334.166666666667
// Output:
// The Garcias in the class are:
// Cesar
// Debra
// Hugo
2. O código anterior neste passo a passo indicou que a pontuação média de classe é de aproximadamente
334. Para produzir uma sequência de Students cuja pontuação total é maior que a média de classe,
juntamente com seus Student ID , você pode usar um tipo anônimo na instrução select :
var studentQuery8 =
from student in students
let x = student.Scores[0] + student.Scores[1] +
student.Scores[2] + student.Scores[3]
where x > averageScore
select new { id = student.ID, score = x };
// Output:
// Student ID: 113, Score: 338
// Student ID: 114, Score: 353
// Student ID: 116, Score: 369
// Student ID: 117, Score: 352
// Student ID: 118, Score: 343
// Student ID: 120, Score: 341
// Student ID: 122, Score: 368
Próximas etapas
Depois que estiver familiarizado com os aspectos básicos de como trabalhar com consultas em C#, você estará
pronto para ler a documentação e exemplos para o tipo específico de provedor LINQ que lhe interessam:
LINQ to SQL
LINQ to DataSet
LINQ to XML (C#)
LINQ to Objects (C#)
Consulte também
LINQ (consulta integrada à linguagem) (C#)
Expressões de consulta LINQ
Visão geral de operadores de consulta padrão
(C#)
23/10/2019 • 6 minutes to read • Edit Online
Os operadores de consulta padrão são os métodos que formam o padrão LINQ. A maioria desses
métodos opera em sequências; neste contexto, uma sequência é um objeto cujo tipo implementa a interface
IEnumerable<T> ou a interface IQueryable<T>. Os operadores de consulta padrão fornecem recursos de
consulta incluindo filtragem, projeção, agregação, classificação e muito mais.
Há dois conjuntos de operadores de consulta padrão LINQ, um que opera em objetos do tipo
IEnumerable<T> e o outro que opera em objetos do tipo IQueryable<T>. Os métodos que compõem a cada
conjunto são os membros estáticos das classes Enumerable e Queryable, respectivamente. Eles são definidos
como métodos de extensão do tipo nos quais operam. Isso significa que eles podem ser chamados usando a
sintaxe de método estático ou sintaxe de método de instância.
Além disso, vários métodos de operador de consulta padrão operam em tipos diferentes daqueles baseados
em IEnumerable<T> ou IQueryable<T>. O tipo Enumerable define dois métodos tais que ambos operam
em objetos do tipo IEnumerable. Esses métodos, Cast<TResult>(IEnumerable) e OfType<TResult>
(IEnumerable), permitem que você habilite uma coleção sem parâmetros ou não genérica, a ser consultada
no padrão LINQ. Eles fazem isso criando uma coleção de objetos fortemente tipada. A classe Queryable
define dois métodos semelhantes, Cast<TResult>(IQueryable) e OfType<TResult>(IQueryable), que operam
em objetos do tipo Queryable.
Os operadores de consulta padrão são diferentes no momento de sua execução, dependendo de se eles
retornam um valor singleton ou uma sequência de valores. Esses métodos que retornam um valor singleton
(por exemplo, Average e Sum) são executados imediatamente. Os métodos que retornam uma sequência
adiam a execução da consulta e retornam um objeto enumerável.
No caso de métodos que operam em coleções na memória, ou seja, os métodos que estendem
IEnumerable<T>, o objeto enumerável retornado captura o argumento que foi passado para o método.
Quando esse objeto é enumerado, a lógica do operador de consulta é empregada e os resultados da consulta
são retornados.
Por outro lado, os métodos que estendem IQueryable<T> não implementam nenhum comportamento de
consulta, mas criam uma árvore de expressão que representa a consulta a ser executada. O processamento
de consulta é tratado pelo objeto IQueryable<T> de origem.
Chamadas para métodos de consulta podem ser encadeadas em uma consulta, o que permite que consultas
se tornem arbitrariamente complexas.
O exemplo de código a seguir demonstra como os operadores de consulta padrão podem ser usados para
obter informações sobre uma sequência.
string sentence = "the quick brown fox jumps over the lazy dog";
// Split the string into individual words to create a collection.
string[] words = sentence.Split(' ');
Seções relacionadas
Os links a seguir levam você a tópicos que fornecem informações adicionais sobre os vários operadores de
consulta padrão com base na funcionalidade.
Classificando dados (C#)
Operações de conjunto (C#)
Filtrando dados (C#)
Operações de quantificador (C#)
Operações de projeção (C#)
Particionando dados (C#)
Operações de junção (C#)
Agrupando dados (C#)
Operações de geração (C#)
Operações de Igualdade (C#)
Operações de elemento (C#)
Convertendo Tipos de Dados (C#)
Operações de concatenação (C#)
Operações de agregação (C#)
Consulte também
Enumerable
Queryable
Introdução a consultas LINQ (C#)
Sintaxe de expressão de consulta para operadores de consulta padrão (C#)
Classificação de operadores de consulta padrão pelo modo de execução (C#)
Métodos de Extensão
Sintaxe de expressão de consulta para operadores de
consulta padrão (C#)
23/10/2019 • 2 minutes to read • Edit Online
Alguns dos operadores de consulta padrão mais usados têm uma sintaxe de palavra-chave de linguagem C#
dedicada que possibilita que eles sejam chamados como parte de uma expressão de consulta. Uma expressão de
consulta é uma maneira diferente e mais legível de expressar uma consulta do que seu equivalente baseado em
método. As cláusulas de expressão de consulta são convertidas em chamadas para os métodos de consulta em
tempo de compilação.
GroupBy group … by
-ou-
group … by … into …
OrderBy<TSource,TKey>(IEnumerable<TSource>, orderby
Func<TSource,TKey>)
(Para obter mais informações, consulte Cláusula orderby.)
Select select
ThenBy<TSource,TKey>(IOrderedEnumerable<TSource>, orderby …, …
Func<TSource,TKey>)
(Para obter mais informações, consulte Cláusula orderby.)
Where where
Consulte também
Enumerable
Queryable
Visão geral de operadores de consulta padrão (C#)
Classificação de operadores de consulta padrão pelo modo de execução (C#)
Classificação de operadores de consulta padrão pelo
modo de execução (C#)
23/10/2019 • 5 minutes to read • Edit Online
As implementações de LINQ to Objects dos métodos de operador de consulta padrão em uma das duas maneiras
principais: imediata ou adiada. Os operadores de consulta que usam a execução adiada podem ser divididos em
mais duas categorias: streaming e não streaming. Se você souber como os operadores de consulta diferentes são
executados, isso poderá ajudá-lo a entender os resultados que serão obtidos de uma determinada consulta. Isso é
especialmente verdadeiro se a fonte de dados está sendo alterada ou se você estiver criando uma consulta sobre
outra consulta. Este tópico classifica os operadores de consulta padrão de acordo com o modo de execução.
Modos de execução
Imediato
A execução imediata significa que a fonte de dados é lida e a operação é realizada no ponto do código em que a
consulta é declarada. Todos os operadores de consulta padrão que retornam um resultado único e não enumerável
são executados imediatamente.
Adiado
A execução adiada significa que a operação não será realizada no ponto do código em que a consulta estiver
declarada. A operação será realizada somente quando a variável de consulta for enumerada, por exemplo, usando
uma instrução foreach . Isso significa que os resultados da execução da consulta dependerão do conteúdo da
fonte de dados quando a consulta for executada em vez de quando a consulta for definida. Se a variável de
consulta for enumerada várias vezes, os resultados poderão ser diferentes a cada vez. Quase todos os operadores
de consulta padrão cujo tipo de retorno é IEnumerable<T> ou IOrderedEnumerable<TElement> executam de
maneira adiada.
Os operadores de consulta que usam a execução adiada podem ser, adicionalmente, classificados como streaming
ou não streaming.
Streaming
Operadores streaming não precisam ler todos os dados de origem antes de gerar elementos. No momento da
execução, um operador streaming realiza sua operação em cada elemento de origem enquanto eles são lidos,
gerando o elemento, se apropriado. Um operador streaming continua a ler os elementos de origem até que um
elemento de resultado possa ser produzido. Isso significa que mais de um elemento de origem poderá ser lido
para produzir um elemento de resultado.
Não streaming
Os operadores não streaming devem ler todos os dados de origem antes de produzirem um elemento de
resultado. Operações como classificação ou agrupamento se enquadram nesta categoria. No momento da
execução, os operadores de consulta não streaming leem todos os dados de origem, colocam-nos em uma
estrutura de dados, realizam a operação e geram os elementos de resultado.
Tabela de classificação
A tabela a seguir classifica cada método de operador de consulta padrão de acordo com o respectivo método de
execução.
NOTE
Se um operador estiver marcado em duas colunas, duas sequências de entrada estarão envolvidas na operação e cada
sequência será avaliada de forma diferente. Nesses casos, a primeira sequência na lista de parâmetros é a que sempre será
avaliada de maneira adiada e em modo streaming.
Aggregate TSource X
All Boolean X
Any Boolean X
AsEnumerable IEnumerable<T> X
Cast IEnumerable<T> X
Concat IEnumerable<T> X
Contains Boolean X
Count Int32 X
DefaultIfEmpty IEnumerable<T> X
Distinct IEnumerable<T> X
ElementAt TSource X
ElementAtOrDefault TSource X
Empty IEnumerable<T> X
Except IEnumerable<T> X X
First TSource X
FirstOrDefault TSource X
GroupBy IEnumerable<T> X
GroupJoin IEnumerable<T> X X
Intersect IEnumerable<T> X X
Join IEnumerable<T> X X
Last TSource X
OPERADOR DE EXECUÇÃO ADIADA DE EXECUÇÃO ADIADA DE
CONSULTA PADRÃO TIPO DE RETORNO EXECUÇÃO IMEDIATA STREAMING NÃO STREAMING
LastOrDefault TSource X
LongCount Int64 X
OfType IEnumerable<T> X
OrderBy IOrderedEnumerable X
<TElement>
OrderByDescending IOrderedEnumerable X
<TElement>
Range IEnumerable<T> X
Repeat IEnumerable<T> X
Reverse IEnumerable<T> X
Select IEnumerable<T> X
SelectMany IEnumerable<T> X
SequenceEqual Boolean X
Single TSource X
SingleOrDefault TSource X
Skip IEnumerable<T> X
SkipWhile IEnumerable<T> X
Take IEnumerable<T> X
TakeWhile IEnumerable<T> X
ThenBy IOrderedEnumerable X
<TElement>
ThenByDescending IOrderedEnumerable X
<TElement>
ToDictionary Dictionary<TKey,TVal X
ue>
ToList IList<T> X
ToLookup ILookup<TKey,TEleme X
nt>
Union IEnumerable<T> X
Where IEnumerable<T> X
Consulte também
Enumerable
Visão geral de operadores de consulta padrão (C#)
Sintaxe de expressão de consulta para operadores de consulta padrão (C#)
LINQ to Objects (C#)
Classificando dados (C#)
04/11/2019 • 4 minutes to read • Edit Online
Uma operação de classificação ordena os elementos de uma sequência com base em um ou mais atributos. O
primeiro critério de classificação executa uma classificação primária dos elementos. Especificando um segundo
critério de classificação, você pode classificar os elementos dentro de cada grupo de classificação primário.
A ilustração a seguir mostra os resultados de uma operação de classificação alfabética em uma sequência de
caracteres:
Os métodos de operador de consulta padrão que classificam dados estão listados na seção a seguir.
Métodos
SINTAXE DE EXPRESSÃO DE
NOME DO MÉTODO DESCRIÇÃO CONSULTA C# MAIS INFORMAÇÕES
Queryable.OrderByDescendi
ng
the
fox
quick
brown
jumps
*/
the
quick
jumps
fox
brown
*/
fox
the
brown
jumps
quick
*/
the
fox
quick
jumps
brown
*/
Consulte também
System.Linq
Visão geral de operadores de consulta padrão (C#)
Cláusula orderby
Como ordenar os resultados de uma cláusula join
Como classificar ou filtrar dados de texto por qualquer palavra ou campo (LINQ ) (C#)
Operações de conjunto (C#)
25/11/2019 • 2 minutes to read • Edit Online
As operações de conjunto na LINQ referem-se a operações de consulta que geram um conjunto de resultados
baseado na presença ou ausência de elementos equivalentes dentro da mesma ou de coleções (ou conjuntos)
separadas.
Os métodos de operador de consulta padrão que executam operações de conjunto estão listados na seção a seguir.
Métodos
SINTAXE DE EXPRESSÃO DE
NOME DO MÉTODO DESCRIÇÃO CONSULTA C# MAIS INFORMAÇÕES
Exceto
A ilustração a seguir mostra o comportamento de Enumerable.Except. A sequência retornada contém apenas os
elementos da primeira sequência de entrada que não estão na segunda sequência de entrada.
Interseção
A ilustração a seguir mostra o comportamento de Enumerable.Intersect. A sequência retornada contém os
elementos que são comuns a ambas as sequências de entrada.
União
A ilustração a seguir descreve uma operação de união em duas sequências de caracteres. A sequência retornada
contém os elementos exclusivos das duas sequências de entrada.
Consulte também
System.Linq
Visão geral de operadores de consulta padrão (C#)
Como combinar e comparar coleções de cadeias de caracteres (C#LINQ ) ()
Como localizar a diferença de conjunto entre duas listas (LINQ ) (C#)
Filtrando dados (C#)
04/11/2019 • 2 minutes to read • Edit Online
A filtragem é a operação de restringir o conjunto de resultados de forma que ele contenha apenas os elementos
correspondentes a uma condição especificada. Ela também é conhecida como seleção.
A ilustração a seguir mostra os resultados da filtragem de uma sequência de caracteres. O predicado para a
operação de filtragem especifica que o caractere deve ser "A".
Os métodos de operador de consulta padrão que realizam a seleção estão listados na seção a seguir.
Métodos
SINTAXE DE EXPRESSÃO DE
NOME DO MÉTODO DESCRIÇÃO CONSULTA C# MAIS INFORMAÇÕES
the
fox
*/
Consulte também
System.Linq
Visão geral de operadores de consulta padrão (C#)
Cláusula where
Como especificar filtros predicados dinamicamente em tempo de execução
Como consultar metadados de um assembly com reflexão (LINQ ) (C#)
Como consultar arquivos com um atributo ou nome especificado (C#)
Como classificar ou filtrar dados de texto por qualquer palavra ou campo (LINQ ) (C#)
Operações de quantificador (C#)
04/11/2019 • 2 minutes to read • Edit Online
As operações de quantificador retornam um valor Boolean que indica se alguns ou todos os elementos em uma
sequência satisfazem uma condição.
A ilustração a seguir mostra duas operações de quantificador diferentes em duas sequências de origem diferentes.
A primeira operação pergunta se um ou mais dos elementos são o caractere 'A' e o resultado é true . A segunda
operação pergunta se todos os elementos são o caractere 'A' e o resultado é true .
Os métodos de operador de consulta padrão que realizam operações de quantificador estão listados na seção a
seguir.
Métodos
SINTAXE DE EXPRESSÃO DE
NOME DO MÉTODO DESCRIÇÃO CONSULTA C# MAIS INFORMAÇÕES
Consulte também
System.Linq
Visão geral de operadores de consulta padrão (C#)
Como especificar filtros predicados dinamicamente em tempo de execução
Como consultar sentenças que contenham um conjunto especificado de palavras (LINQ ) (C#)
Operações de projeção (C#)
23/10/2019 • 5 minutes to read • Edit Online
Projeção refere-se à operação de transformar um objeto em um novo formulário que geralmente consiste apenas
nas propriedades que serão usadas posteriormente. Usando a projeção, você pode construir um novo tipo que é
criado de cada objeto. É possível projetar uma propriedade e executar uma função matemática nela. Também é
possível projetar o objeto original sem alterá-lo.
Os métodos de operador de consulta padrão que realizam a projeção estão listados na seção a seguir.
Métodos
SINTAXE DE EXPRESSÃO DE
NOME DO MÉTODO DESCRIÇÃO CONSULTA C# MAIS INFORMAÇÕES
a
a
a
d
*/
SelectMany
O exemplo a seguir usa várias cláusulas from para projetar cada palavra de cada cadeia de caracteres em uma
lista de cadeias de caracteres.
List<string> phrases = new List<string>() { "an apple a day", "the quick brown fox" };
an
apple
a
day
the
quick
brown
fox
*/
Esta ilustração mostra como SelectMany() concatena a sequência intermediária de matrizes em um valor de
resultado final que contém cada valor de cada matriz intermediária.
Exemplo de código
O exemplo a seguir compara o comportamento de Select() e de SelectMany() . O código cria um "buquê" de
flores usando os dois primeiros itens de cada lista de nomes de flor na coleção de origem. Neste exemplo, o "valor
único" que a função de transformação Select<TSource,TResult>(IEnumerable<TSource>,
Func<TSource,TResult>) usa é uma coleção de valores. Isso requer o loop foreach extra para enumerar cada
cadeia de caracteres em cada subsequência.
class Bouquet
{
public List<string> Flowers { get; set; }
}
Consulte também
System.Linq
Visão geral de operadores de consulta padrão (C#)
Cláusula select
Como: Popular coleções de objetos de várias fontes (LINQ ) (C#)
Como: Dividir um arquivo em vários arquivos usando grupos (LINQ ) (C#)
Particionando dados (C#)
23/10/2019 • 2 minutes to read • Edit Online
Particionamento em LINQ refere-se à operação de dividir uma sequência de entrada em duas seções sem
reorganizar os elementos e, depois, retornar uma das seções.
A ilustração a seguir mostra os resultados de três operações de particionamento diferentes em uma sequência de
caracteres. A primeira operação retorna os três primeiros elementos na sequência. A segunda operação ignora os
três primeiros elementos e retorna os elementos restantes. A terceira operação ignora os dois primeiros elementos
na sequência e retorna os três elementos seguintes.
Os métodos de operador de consulta padrão que particionam sequências estão listados na seção a seguir.
Operadores
SINTAXE DE EXPRESSÃO DE
NOME DO OPERADOR DESCRIÇÃO CONSULTA C# MAIS INFORMAÇÕES
Consulte também
System.Linq
Visão geral de operadores de consulta padrão (C#)
Operações de junção (C#)
04/11/2019 • 4 minutes to read • Edit Online
Uma junção de duas fontes de dados é a associação de objetos em uma fonte de dados, com objetos que
compartilham um atributo comum em outra fonte de dados.
A junção é uma operação importante em consultas que têm como destino fontes de dados cujas relações entre si
não podem ser seguidas diretamente. Na programação orientada a objeto, isso pode significar uma correlação
entre objetos que não são modelados, como a direção retroativa de uma relação unidirecional. Um exemplo de
uma relação unidirecional é uma classe Cliente que tem uma propriedade do tipo Cidade, mas a classe Cidade
ainda não tem uma propriedade que é uma coleção de objetos Cliente. Se você tem uma lista de objetos Cidade e
você quer encontrar todos os clientes em cada cidade, você pode usar uma operação de junção para encontrá-los.
Os métodos de junção fornecidos na estrutura do LINQ são Join e GroupJoin. Esses métodos executam junção
por igualdade ou junções que correspondem duas fontes de dados com base na igualdade de suas chaves. (Para
comparação, o Transact-SQL dá suporte a operadores de junção diferentes de ' Equals ', por exemplo, o operador '
less than '.) Em termos de banco de dados relacional, Join implementa uma junção interna, um tipo de junção em
que somente os objetos que têm uma correspondência no outro conjunto de dado são retornados. O método
GroupJoin não tem equivalente direto em termos de banco de dados relacional, mas ele implementa um
superconjunto de junções internas e junções externas esquerdas. Uma junção externa esquerda é uma junção que
retorna cada elemento da primeira (esquerda) fonte de dados, mesmo que ele não tenha elementos
correlacionados na outra fonte de dados.
A ilustração a seguir mostra uma visão conceitual de dois conjuntos e os elementos dentro desses conjuntos que
estão incluídos em uma junção interna ou externa à esquerda.
Métodos
SINTAXE DE EXPRESSÃO DE
NOME DO MÉTODO DESCRIÇÃO CONSULTA C# MAIS INFORMAÇÕES
O agrupamento refere-se à operação de colocação de dados em grupos, de modo que os elementos em cada
grupo compartilhem um atributo comum.
A ilustração a seguir mostra os resultados do agrupamento de uma sequência de caracteres. A chave para cada
grupo é o caractere.
Os métodos do operador de consulta padrão que agrupam elementos de dados estão listados na seção a seguir.
Métodos
SINTAXE DE EXPRESSÃO DE
NOME DO MÉTODO DESCRIÇÃO CONSULTA C# MAIS INFORMAÇÕES
Odd numbers:
35
3987
199
329
Even numbers:
44
200
84
4
446
208
*/
Consulte também
System.Linq
Visão geral de operadores de consulta padrão (C#)
Cláusula group
Como criar um grupo aninhado
Como agrupar arquivos por extensão (LINQ ) (C#)
Como agrupar resultados de consultas
Como executar uma subconsulta em uma operação de agrupamento
Como dividir um arquivo em vários arquivos usando grupos (LINQ ) (C#)
Operações de geração (C#)
23/10/2019 • 2 minutes to read • Edit Online
Métodos
SINTAXE DE EXPRESSÃO DE
NOME DO MÉTODO DESCRIÇÃO CONSULTA C# MAIS INFORMAÇÕES
Consulte também
System.Linq
Visão geral de operadores de consulta padrão (C#)
Operações de Igualdade (C#)
25/11/2019 • 2 minutes to read • Edit Online
Duas sequências cujos elementos correspondentes são iguais e que têm o mesmo número de elementos são
consideradas iguais.
Métodos
SINTAXE DE EXPRESSÃO DE
NOME DO MÉTODO DESCRIÇÃO CONSULTA C# MAIS INFORMAÇÕES
Consulte também
System.Linq
Visão geral de operadores de consulta padrão (C#)
Como comparar o conteúdo de duas pastas (LINQ ) (C#)
Operações de elemento (C#)
23/10/2019 • 2 minutes to read • Edit Online
Métodos
SINTAXE DE EXPRESSÃO DE
NOME DO MÉTODO DESCRIÇÃO CONSULTA C# MAIS INFORMAÇÕES
Consulte também
System.Linq
Visão geral de operadores de consulta padrão (C#)
Como: Consultar o maior arquivo ou arquivos em uma árvore de diretório (LINQ ) (C#)
Convertendo Tipos de Dados (C#)
04/11/2019 • 3 minutes to read • Edit Online
Métodos
A tabela a seguir lista os métodos de operador de consulta padrão que realizam conversões de tipo de dados.
Os métodos de conversão nesta tabela cujos nomes começam com "As" alteram o tipo estático da coleção de
origem, mas não a enumeram. Os métodos cujos nomes começam com "To" enumeram a coleção de origem e
colocam os itens na coleção de tipo correspondente.
SINTAXE DE EXPRESSÃO DE
NOME DO MÉTODO DESCRIÇÃO CONSULTA C# MAIS INFORMAÇÕES
class Plant
{
public string Name { get; set; }
}
Os métodos de operador de consulta padrão que realizam a concatenação estão listados na seção a seguir.
Métodos
SINTAXE DE EXPRESSÃO DE
NOME DO MÉTODO DESCRIÇÃO CONSULTA C# MAIS INFORMAÇÕES
Consulte também
System.Linq
Visão geral de operadores de consulta padrão (C#)
Como combinar e comparar coleções de cadeias de caracteres (C#LINQ ) ()
Operações de agregação (C#)
25/11/2019 • 2 minutes to read • Edit Online
Uma operação de agregação computa um único valor de uma coleção de valores. Um exemplo de uma operação
de agregação é o cálculo da temperatura média diária dos valores válidos de temperatura diária de um mês.
A ilustração a seguir mostra os resultados de duas operações de agregação diferentes em uma sequência de
números. A primeira operação soma os números. A segunda operação retorna o valor máximo na sequência.
Os métodos de operador de consulta padrão que realizam operações de agregação são listados na seção a seguir.
Métodos
SINTAXE DE EXPRESSÃO DE
NOME DO MÉTODO DESCRIÇÃO CONSULTA C# MAIS INFORMAÇÕES
Consulte também
System.Linq
Visão geral de operadores de consulta padrão (C#)
Como computar valores de coluna em um arquivo de texto CSV (C#LINQ ) ()
Como consultar o maior arquivo ou arquivos em uma árvore de diretório (LINQ ) (C#)
Como consultar o número total de bytes em um conjunto de pastas (LINQ ) (C#)
LINQ to Objects (C#)
25/11/2019 • 3 minutes to read • Edit Online
O termo "LINQ to Objects" refere-se ao uso de consultas LINQ com qualquer coleção IEnumerable ou
IEnumerable<T> diretamente, sem o uso de uma API ou provedor LINQ intermediário como o LINQ to SQL
ou LINQ to XML. Você pode usar o LINQ para consultar qualquer coleção enumerável como List<T>, Array ou
Dictionary<TKey,TValue>. A coleção pode ser definida pelo usuário ou pode ser devolvida por uma API do
.NET Framework.
Basicamente, o LINQ to Objects representa uma nova abordagem às coleções. Na forma antiga, você precisava
escrever loops foreach complexos que especificavam como recuperar dados de uma coleção. Na abordagem
da LINQ, você escreve o código declarativo que descreve o que você deseja recuperar.
Além disso, as consultas LINQ oferecem três principais vantagens sobre os loops foreach tradicionais:
1. Elas são mais concisas e legíveis, especialmente quando você filtra várias condições.
2. Elas fornecem poderosos recursos de filtragem, ordenação e agrupamento com um mínimo de código
do aplicativo.
3. Elas podem ser movidas para outras fontes de dados com pouca ou nenhuma modificação.
Em geral, quanto mais complexa a operação que você deseja executar sobre os dados, maior benefício você
perceberá usando consultas LINQs em vez de técnicas tradicionais de iteração.
O objetivo desta seção é demonstrar a abordagem LINQ com alguns exemplos selecionados. Não pretendemos
que ela seja detalhada.
Nesta seção
LINQ e cadeias de caracteres (C#)
Explica como a LINQ pode ser usada para consultar e transformar cadeias de caracteres e coleções de cadeias
de caracteres. Também inclui links para tópicos que demonstram esses princípios.
LINQ e reflexão (C#)
Contém um link para um exemplo que demonstra como a LINQ usa a reflexão.
LINQ e diretórios de arquivos (C#)
Explica como a LINQ pode ser usada para interagir com sistemas de arquivos. Também inclui links para tópicos
que demonstram esses conceitos.
Como consultar um ArrayList com LINQ (C#)
Demonstra como consultar um ArrayList no C#.
Como adicionar métodos personalizados para consultas LINQ (C#)
Explica como estender o conjunto de métodos que você pode usar para consultas LINQ, adicionando os
métodos de extensão à interface IEnumerable<T>.
LINQ (consulta integrada à linguagem) (C#)
Fornece links para tópicos que explicam a LINQ e fornecem exemplos de código que realizam consultas.
LINQ e cadeias de caracteres (C#)
23/11/2019 • 5 minutes to read • Edit Online
A LINQ pode ser usada para consultar e transformar as cadeias de caracteres e coleções de cadeias de
caracteres. Ele pode ser especialmente útil com os dados semiestruturados em arquivos de texto. Consultas
LINQ podem ser combinadas com expressões regulares e funções de cadeia de caracteres tradicionais. Por
exemplo, você pode usar o método String.Split ou Regex.Split para criar uma matriz de cadeias de caracteres que
você pode consultar ou modificar usando o LINQ. Você pode usar o método Regex.IsMatch na cláusula where
de uma consulta LINQ. E você pode usar o LINQ para consultar ou modificar os resultados de MatchCollection
retornados por uma expressão regular.
Você também pode usar as técnicas descritas nessa seção para transformar dados de texto semiestruturados em
XML. Para obter mais informações, consulte Como gerar um XML de arquivos CSV.
Os exemplos nesta seção se enquadram em duas categorias:
Consulte também
LINQ (consulta integrada à linguagem) (C#)
Como gerar um XML de arquivos CSV
Como contar ocorrências de uma palavra em uma
cadeia de caracteres (LINQ)C#()
25/11/2019 • 2 minutes to read • Edit Online
Este exemplo mostra como usar uma consulta LINQ para contar as ocorrências de uma palavra especificada em
uma cadeia de caracteres. Observe que para executar a contagem, primeiro o método Split é chamado para criar
uma matriz de palavras. Há um custo de desempenho para o método Split. Se for a única operação na cadeia de
caracteres for contar as palavras, você deverá considerar o uso dos métodos Matches ou IndexOf em vez dele. No
entanto, se o desempenho não for um problema crítico ou se você já tiver dividido a sentença para executar outros
tipos de consulta nela, faz sentido usar LINQ para contar as palavras ou frases também.
Exemplo
class CountWords
{
static void Main()
{
string text = @"Historically, the world of data and the world of objects" +
@" have not been well integrated. Programmers work in C# or Visual Basic" +
@" and also in SQL or XQuery. On the one side are concepts such as classes," +
@" objects, fields, inheritance, and .NET Framework APIs. On the other side" +
@" are tables, columns, rows, nodes, and separate languages for dealing with" +
@" them. Data types often require translation between the two worlds; there are" +
@" different standard functions. Because the object world has no notion of query, a" +
@" query can only be represented as a string without compile-time type checking or" +
@" IntelliSense support in the IDE. Transferring data from SQL tables or XML trees to" +
@" objects in memory is often tedious and error-prone.";
Compilando o código
Criar um projeto de aplicativo de console em C# com diretivas using para os namespaces System.Linq e
System.IO.
Consulte também
LINQ e cadeias de caracteres (C#)
Como: Consultar sentenças que contêm um conjunto
especificado de palavras (LINQ) (C#)
31/10/2019 • 3 minutes to read • Edit Online
Este exemplo mostra como localizar frases em um arquivo de texto que contenham correspondências para cada
conjunto de palavras especificado. Embora a matriz de termos de pesquisa esteja embutida em código neste
exemplo, ela também poderia ser populada dinamicamente em tempo de execução. Neste exemplo, a consulta
retorna as frases que contêm as palavras "Historically", "data" e "integrated".
Exemplo
class FindSentences
{
static void Main()
{
string text = @"Historically, the world of data and the world of objects " +
@"have not been well integrated. Programmers work in C# or Visual Basic " +
@"and also in SQL or XQuery. On the one side are concepts such as classes, " +
@"objects, fields, inheritance, and .NET Framework APIs. On the other side " +
@"are tables, columns, rows, nodes, and separate languages for dealing with " +
@"them. Data types often require translation between the two worlds; there are " +
@"different standard functions. Because the object world has no notion of query, a " +
@"query can only be represented as a string without compile-time type checking or " +
@"IntelliSense support in the IDE. Transferring data from SQL tables or XML trees to " +
@"objects in memory is often tedious and error-prone.";
// Define the search terms. This list could also be dynamically populated at runtime.
string[] wordsToMatch = { "Historically", "data", "integrated" };
// Find sentences that contain all the terms in the wordsToMatch array.
// Note that the number of terms to match is not specified at compile time.
var sentenceQuery = from sentence in sentences
let w = sentence.Split(new char[] { '.', '?', '!', ' ', ';', ':', ',' },
StringSplitOptions.RemoveEmptyEntries)
where w.Distinct().Intersect(wordsToMatch).Count() == wordsToMatch.Count()
select sentence;
A consulta funciona primeiro dividindo o texto em frases e, em seguida, dividindo as sentenças em uma matriz de
cadeias de caracteres que contêm cada palavra. Para cada uma dessas matrizes, o método Distinct remove todas
as palavras duplicadas e, em seguida, a consulta executa uma operação Intersect na matriz de palavras e na matriz
wordsToMatch . Se a contagem da interseção for igual à contagem da matriz wordsToMatch , todas as palavras foram
encontradas nas palavras e a frase original será retornada.
Na chamada para Split, as marcas de pontuação são usadas como separadores para removê-las da cadeia de
caracteres. Se não fizer isso, por exemplo, você poderia ter uma cadeia de caracteres "Historically" que não
corresponderia a "Historically" na matriz wordsToMatch . Talvez você precise usar separadores adicionais,
dependendo dos tipos de pontuação encontrados no texto de origem.
Compilando o código
Criar um projeto de aplicativo de console em C# com diretivas using para os namespaces System.Linq e
System.IO.
Consulte também
LINQ e cadeias de caracteres (C#)
Como consultar caracteres em uma cadeia de
caracteres (LINQ) (C#)
25/11/2019 • 2 minutes to read • Edit Online
Já que a classe String implementa a interface IEnumerable<T> genérica, qualquer cadeia de caracteres pode ser
consultada como uma sequência de caracteres. No entanto, esse não é um uso comum da LINQ. Para operações
de correspondência de padrões complexas, use a classe Regex.
Exemplo
O exemplo a seguir consulta uma cadeia de caracteres para determinar quantos dígitos numéricos ela contém.
Observe que a consulta é "reutilizada" depois que é executada pela primeira vez. Isso é possível porque a consulta
em si não armazena nenhum resultado real.
class QueryAString
{
static void Main()
{
string aString = "ABCDE99F-J74-12-89A";
Compilando o código
Criar um projeto de aplicativo de console em C# com diretivas using para os namespaces System.Linq e
System.IO.
Consulte também
LINQ e cadeias de caracteres (C#)
Como combinar consultas LINQ com expressões regulares (C#)
Como combinar consultas LINQ com expressões
regulares (C#)
25/11/2019 • 3 minutes to read • Edit Online
Este exemplo mostra como usar a classe Regex para criar uma expressão regular para correspondências mais
complexas em cadeias de texto. A consulta LINQ torna fácil a aplicação de filtro exatamente nos arquivos que você
deseja pesquisar com a expressão regular e formatar os resultados.
Exemplo
class QueryWithRegEx
{
public static void Main()
{
// Modify this path as necessary so that it accesses your version of Visual Studio.
string startFolder = @"C:\Program Files (x86)\Microsoft Visual Studio 14.0\";
// One of the following paths may be more appropriate on your computer.
//string startFolder = @"C:\Program Files (x86)\Microsoft Visual Studio\2017\";
Observe que também é possível consultar o objeto MatchCollection retornado por uma pesquisa RegEx . Neste
exemplo, apenas o valor de cada correspondência é produzido nos resultados. No entanto, também é possível
usar a LINQ para executar todos os tipos de filtragem, classificação e agrupamento nessa coleção. Como
MatchCollection é uma coleção IEnumerable não genérica, é necessário declarar explicitamente o tipo da variável
de intervalo na consulta.
Compilando o código
Criar um projeto de aplicativo de console em C# com diretivas using para os namespaces System.Linq e
System.IO.
Consulte também
LINQ e cadeias de caracteres (C#)
LINQ e diretórios de arquivos (C#)
Como localizar a diferença de conjunto entre duas
listas (LINQ) (C#)
25/11/2019 • 2 minutes to read • Edit Online
Este exemplo mostra como usar o LINQ para comparar duas listas de cadeias de caracteres e retornar as linhas
que estão no names1.txt, mas não no names2.txt.
Para criar os arquivos de dados
1. Copie names1. txt e names2. txt para a pasta da solução, conforme mostrado em como combinar e comparar
as coleções de cadeiasC#de caracteres (LINQ ) ().
Exemplo
class CompareLists
{
static void Main()
{
// Create the IEnumerable data sources.
string[] names1 = System.IO.File.ReadAllLines(@"../../../names1.txt");
string[] names2 = System.IO.File.ReadAllLines(@"../../../names2.txt");
// Create the query. Note that method syntax must be used here.
IEnumerable<string> differenceQuery =
names1.Except(names2);
Alguns tipos de operações de consulta em C#, tais como Except, Distinct, Union e Concat, só podem ser expressas
em sintaxe baseada em método.
Compilando o código
Criar um projeto de aplicativo de console em C# com diretivas using para os namespaces System.Linq e
System.IO.
Consulte também
LINQ e cadeias de caracteres (C#)
Como: Classificar ou filtrar dados de texto por
qualquer palavra ou campo (LINQ) (C#)
31/10/2019 • 2 minutes to read • Edit Online
O exemplo a seguir mostra como classificar linhas de texto estruturado, como valores separados por vírgulas, por
qualquer campo na linha. O campo pode ser especificado dinamicamente em tempo de execução. Suponha que os
campos em scores.csv representam o número de ID do aluno, seguido por uma série de quatro resultados de
teste.
Para criar um arquivo que contém dados
1. Copie os dados de scores.csv do tópico Como: Unir o conteúdo de arquivos diferentes (LINQ ) (C#) e salve-o
na pasta da solução.
Exemplo
public class SortLines
{
static void Main()
{
// Create an IEnumerable data source
string[] scores = System.IO.File.ReadAllLines(@"../../../scores.csv");
return scoreQuery;
}
}
/* Output (if sortField == 1):
Sorted highest to lowest by field [1]:
116, 99, 86, 90, 94
120, 99, 82, 81, 79
111, 97, 92, 81, 60
114, 97, 89, 85, 82
121, 96, 85, 91, 60
122, 94, 92, 91, 91
117, 93, 92, 80, 87
118, 92, 90, 83, 78
113, 88, 94, 65, 91
112, 75, 84, 91, 39
119, 68, 79, 88, 92
115, 35, 72, 91, 70
*/
Este exemplo também demonstra como retornar uma variável de consulta de um método.
Compilando o código
Criar um projeto de aplicativo de console em C# com diretivas using para os namespaces System.Linq e
System.IO.
Consulte também
LINQ e cadeias de caracteres (C#)
Como: Reordenar os campos de um arquivo
delimitado (LINQ) (C#)
23/10/2019 • 3 minutes to read • Edit Online
Um CSV (arquivo de valores separados por vírgula) é um arquivo de texto que é frequentemente usado para
armazenar dados de planilha ou outros dados de tabela que são representados por linhas e colunas. Ao usar o
método Split para separar os campos, é muito fácil consultar e manipular arquivos CSV usando LINQ. Na
verdade, a mesma técnica pode ser usada para reordenar as partes de qualquer linha estruturada de texto. Ela não
é limitada a arquivos CSV.
No exemplo a seguir, suponha que as três colunas representam o "sobrenome", o "nome" e a "ID" dos alunos.Os
campos estão em ordem alfabética com base nos sobrenomes dos alunos. A consulta gera uma nova sequência, na
qual a coluna ID é exibida em primeiro, seguida por uma segunda coluna que combina o nome e o sobrenome do
aluno. As linhas são reordenadas acordo com o campo ID. Os resultados são salvos em um novo arquivo e os
dados originais não são modificados.
Para criar o arquivo de dados
1. Copie as seguintes linhas em um arquivo de texto sem formatação denominado spreadsheet1.csv. Salve o
arquivo na pasta do seu projeto.
Adams,Terry,120
Fakhouri,Fadi,116
Feng,Hanying,117
Garcia,Cesar,114
Garcia,Debra,115
Garcia,Hugo,118
Mortensen,Sven,113
O'Donnell,Claire,112
Omelchenko,Svetlana,111
Tucker,Lance,119
Tucker,Michael,122
Zabokritski,Eugene,121
Exemplo
class CSVFiles
{
static void Main(string[] args)
{
// Create the IEnumerable data source
string[] lines = System.IO.File.ReadAllLines(@"../../../spreadsheet1.csv");
// Execute the query and write out the new file. Note that WriteAllLines
// takes a string[], so ToArray is called on the query.
System.IO.File.WriteAllLines(@"../../../spreadsheet2.csv", query.ToArray());
Compilando o código
Criar um projeto de aplicativo de console em C# com diretivas using para os namespaces System.Linq e
System.IO.
Consulte também
LINQ e cadeias de caracteres (C#)
LINQ e diretórios de arquivos (C#)
Como: Gerar um XML com base em arquivos CSV (C#)
Como combinar e comparar coleções de cadeias de
caracteres (C#LINQ) ()
25/11/2019 • 3 minutes to read • Edit Online
Este exemplo mostra como mesclar arquivos que contêm linhas de texto e, em seguida, classificar os resultados.
Especificamente, mostra como executar uma concatenação, uma união e uma interseção simples nos dois
conjuntos de linhas de texto.
Para configurar o projeto e os arquivos de texto
1. Copie esses nomes em um arquivo de texto chamado names1.txt e salve-o na sua pasta do projeto:
Bankov, Peter
Holm, Michael
Garcia, Hugo
Potra, Cristina
Noriega, Fabricio
Aw, Kam Foo
Beebe, Ann
Toyoshima, Tim
Guy, Wey Yuan
Garcia, Debra
2. Copie esses nomes em um arquivo de texto chamado names2.txt e salve-o na sua pasta do projeto.
Observe que os dois arquivos tem alguns nomes em comum.
Liu, Jinghao
Bankov, Peter
Holm, Michael
Garcia, Hugo
Beebe, Ann
Gilchrist, Beth
Myrcha, Jacek
Giakoumakis, Leo
McLin, Nkenge
El Yassir, Mehdi
Exemplo
class MergeStrings
{
static void Main(string[] args)
{
//Put text files in your solution folder
string[] fileA = System.IO.File.ReadAllLines(@"../../../names1.txt");
string[] fileB = System.IO.File.ReadAllLines(@"../../../names2.txt");
IEnumerable<String> tempQuery1 =
from name in fileA
let n = name.Split(',')
where n[0] == nameMatch
select name;
IEnumerable<string> tempQuery2 =
from name2 in fileB
let n2 = name2.Split(',')
where n2[0] == nameMatch
select name2;
IEnumerable<string> nameMatchQuery =
tempQuery1.Concat(tempQuery2).OrderBy(s => s);
OutputQueryResults(nameMatchQuery, $"Concat based on partial name match \"{nameMatch}\":");
Compilando o código
Criar um projeto de aplicativo de console em C# com diretivas using para os namespaces System.Linq e
System.IO.
Consulte também
LINQ e cadeias de caracteres (C#)
LINQ e diretórios de arquivos (C#)
Como: Popular coleções de objetos de várias fontes
(LINQ) (C#)
23/10/2019 • 5 minutes to read • Edit Online
Este exemplo mostra como mesclar dados de diferentes fontes em uma sequência de novos tipos.
NOTE
Não tente unir dados na memória ou dados no sistema de arquivos com os dados que ainda estão em um banco de dados.
Essas junções entre domínios podem gerar resultados indefinidos, devido às diferentes formas em que as operações de
junção podem ser definidas para consultas de banco de dados e outros tipos de fontes. Além disso, há um risco de que essa
operação possa causar uma exceção de falta de memória, se a quantidade de dados no banco de dados for grande o
suficiente. Para unir dados de um banco de dados com os dados na memória, primeiro chame ToList ou ToArray na
consulta de banco de dados e, em seguida, realize a junção na coleção retornada.
Exemplo
O exemplo a seguir mostra como usar um tipo nomeado Student para armazenar dados mesclados, de duas
coleções na memória de cadeias de caracteres, que simulam dados de planilha no formato .csv. A primeira coleção
de cadeias de caracteres representa os nomes e as IDs dos alunos e a segunda coleção, representa a ID do aluno
(na primeira coluna) e quatro pontuações de exames. A ID é usada como a chave estrangeira.
using System;
using System.Collections.Generic;
using System.Linq;
class Student
{
public string FirstName { get; set; }
public string LastName { get; set; }
public int ID { get; set; }
public List<int> ExamScores { get; set; }
}
class PopulateCollection
{
static void Main()
{
// These data files are defined in How to: Join Content from
// Dissimilar Files (LINQ).
Na cláusula select, um inicializador de objeto é usado para instanciar cada novo objeto Student , usando os dados
das duas fontes.
Se você não tiver que armazenar os resultados de uma consulta, os tipos anônimos poderão ser mais
convenientes que os tipos nomeados. Os tipos nomeados são necessários se você passa os resultados da consulta
para fora do método em que a consulta é executada. O exemplo a seguir realiza a mesma tarefa do exemplo
anterior, mas usa tipos anônimos em vez de tipos nomeados:
// Merge the data sources by using an anonymous type.
// Note the dynamic creation of a list of ints for the
// ExamScores member. We skip 1 because the first string
// in the array is the student ID, not an exam score.
var queryNamesScores2 =
from nameLine in names
let splitName = nameLine.Split(',')
from scoreLine in scores
let splitScoreLine = scoreLine.Split(',')
where Convert.ToInt32(splitName[2]) == Convert.ToInt32(splitScoreLine[0])
select new
{
First = splitName[0],
Last = splitName[1],
ExamScores = (from scoreAsText in splitScoreLine.Skip(1)
select Convert.ToInt32(scoreAsText))
.ToList()
};
Consulte também
LINQ e cadeias de caracteres (C#)
Inicializadores de objeto e coleção
Tipos Anônimos
Como: Dividir um arquivo em vários arquivos usando
grupos (LINQ) (C#)
23/10/2019 • 2 minutes to read • Edit Online
Este exemplo mostra uma maneira de mesclar o conteúdo de dois arquivos e, em seguida, criar um conjunto de
novos arquivos que organizam os dados em uma nova forma.
Para criar os arquivos de dados
1. Copie esses nomes em um arquivo de texto chamado names1.txt e salve-o na sua pasta do projeto:
Bankov, Peter
Holm, Michael
Garcia, Hugo
Potra, Cristina
Noriega, Fabricio
Aw, Kam Foo
Beebe, Ann
Toyoshima, Tim
Guy, Wey Yuan
Garcia, Debra
2. Copie estes nomes em um arquivo de texto chamado names2.txt e salve-o na pasta do projeto: Observe
que os dois arquivos tem alguns nomes em comum.
Liu, Jinghao
Bankov, Peter
Holm, Michael
Garcia, Hugo
Beebe, Ann
Gilchrist, Beth
Myrcha, Jacek
Giakoumakis, Leo
McLin, Nkenge
El Yassir, Mehdi
Exemplo
class SplitWithGroups
{
static void Main()
{
string[] fileA = System.IO.File.ReadAllLines(@"../../../names1.txt");
string[] fileB = System.IO.File.ReadAllLines(@"../../../names2.txt");
// Output to display.
Console.WriteLine(g.Key);
// Write file.
using (System.IO.StreamWriter sw = new System.IO.StreamWriter(fileName))
{
foreach (var item in g)
{
sw.WriteLine(item);
// Output to console for example purposes.
Console.WriteLine(" {0}", item);
}
}
}
// Keep console window open in debug mode.
Console.WriteLine("Files have been written. Press any key to exit");
Console.ReadKey();
}
}
/* Output:
A
Aw, Kam Foo
B
Bankov, Peter
Beebe, Ann
E
El Yassir, Mehdi
G
Garcia, Hugo
Guy, Wey Yuan
Garcia, Debra
Gilchrist, Beth
Giakoumakis, Leo
H
Holm, Michael
L
Liu, Jinghao
M
Myrcha, Jacek
McLin, Nkenge
N
Noriega, Fabricio
P
Potra, Cristina
T
Toyoshima, Tim
*/
O programa grava um arquivo separado para cada grupo na mesma pasta que os arquivos de dados.
Compilando o código
Criar um projeto de aplicativo de console em C# com diretivas using para os namespaces System.Linq e
System.IO.
Consulte também
LINQ e cadeias de caracteres (C#)
LINQ e diretórios de arquivos (C#)
Como: Unir o conteúdo de arquivos diferentes
(LINQ) (C#)
23/10/2019 • 3 minutes to read • Edit Online
Este exemplo mostra como unir dados de dois arquivos delimitados por vírgulas que compartilham um valor
comum que é usado como uma chave correspondente. Essa técnica pode ser útil se você precisa combinar dados
de duas planilhas ou de uma planilha e um arquivo com outro formato, em um novo arquivo. Você pode
modificar o exemplo para funcionar com qualquer tipo de texto estruturado.
2. Copie as seguintes linhas para um arquivo chamado names.csv e salve-o na sua pasta do projeto. O
arquivo representa uma planilha que contém o sobrenome, o nome e a ID do aluno.
Omelchenko,Svetlana,111
O'Donnell,Claire,112
Mortensen,Sven,113
Garcia,Cesar,114
Garcia,Debra,115
Fakhouri,Fadi,116
Feng,Hanying,117
Garcia,Hugo,118
Tucker,Lance,119
Adams,Terry,120
Zabokritski,Eugene,121
Tucker,Michael,122
Exemplo
using System;
using System.Collections.Generic;
using System.Linq;
class JoinStrings
{
static void Main()
{
{
// Join content from dissimilar files that contain
// related information. File names.csv contains the student
// name plus an ID number. File scores.csv contains the ID
// and a set of four test scores. The following query joins
// the scores to the student names by using ID as a
// matching key.
Consulte também
LINQ e cadeias de caracteres (C#)
LINQ e diretórios de arquivos (C#)
Como computar valores de coluna em um arquivo de
texto CSV (C#LINQ) ()
25/11/2019 • 4 minutes to read • Edit Online
Este exemplo mostra como executar cálculos de agregação, como soma, média, mín. e máx. nas colunas de um
arquivo .csv. Os princípios de exemplo mostrados aqui podem ser aplicados a outros tipos de texto estruturado.
Exemplo
class SumColumns
{
static void Main(string[] args)
{
string[] lines = System.IO.File.ReadAllLines(@"../../../scores.csv");
// Spreadsheet format:
// Student ID Exam#1 Exam#2 Exam#3 Exam#4
// 111, 97, 92, 81, 60
A consulta funciona usando o método Split para converter cada linha de texto em uma matriz. Cada elemento da
matriz representa uma coluna. Por fim, o texto em cada coluna é convertido em sua representação numérica. Se o
arquivo for um arquivo separado por tabulações, é só atualizar o argumento no método Split para \t .
Compilando o código
Criar um projeto de aplicativo de console em C# com diretivas using para os namespaces System.Linq e
System.IO.
Consulte também
LINQ e cadeias de caracteres (C#)
LINQ e diretórios de arquivos (C#)
2 minutes to read
Como: Consultar metadados de um assembly com a
reflexão (LINQ) (C#)
23/10/2019 • 2 minutes to read • Edit Online
As APIs de reflexão da biblioteca de classes .NET Framework podem ser usadas para examinar os metadados no
assembly .NET e criar coleções de tipos, membros de tipo, parâmetros e assim por diante que estão nesse
assembly. Como essas coleções dão suporte à interface IEnumerable<T> genéricas, elas podem ser consultadas
usando LINQ.
O exemplo a seguir mostra como o LINQ pode ser usado com a reflexão para recuperar metadados específicos
sobre os métodos que correspondem a um critério de pesquisa especificado. Nesse caso, a consulta localizará os
nomes de todos os métodos no assembly que retornam tipos enumeráveis como matrizes.
Exemplo
using System;
using System.Linq;
using System.Reflection;
class ReflectionHowTO
{
static void Main()
{
Assembly assembly = Assembly.Load("System.Core, Version=3.5.0.0, Culture=neutral, PublicKeyToken=
b77a5c561934e089");
var pubTypesQuery = from type in assembly.GetTypes()
where type.IsPublic
from method in type.GetMethods()
where method.ReturnType.IsArray == true
|| ( method.ReturnType.GetInterface(
typeof(System.Collections.Generic.IEnumerable<>).FullName ) != null
&& method.ReturnType.FullName != "System.String" )
group method.ToString() by type.ToString();
O exemplo usa o método Assembly.GetTypes para retornar uma matriz de tipos no assembly especificado. O filtro
where é aplicado para que apenas tipos públicos sejam retornados. Para cada tipo de público, uma subconsulta é
gerada usando a matriz MethodInfo que é retornada da chamada Type.GetMethods. Esses resultados são filtrados
para retornar apenas os métodos cujo tipo de retorno é uma matriz ou um tipo que implementa IEnumerable<T>.
Por fim, esses resultados são agrupados usando o nome do tipo como uma chave.
Consulte também
LINQ to Objects (C#)
LINQ e diretórios de arquivos (C#)
25/11/2019 • 4 minutes to read • Edit Online
Muitas operações do sistema de arquivos são essencialmente consultas e, portanto, são ideais para a
abordagem do LINQ.
Observe que as consultas nesta seção são não destrutivas. Elas não são usadas para alterar o conteúdo dos
arquivos ou pastas originais. Isso segue a regra de que consultas não devem causar efeitos colaterais. Em geral,
qualquer código (incluindo consultas que executam operadores criar / atualizar / excluir) que modifique dados
de origem deve ser mantido separado do código que apenas consulta os dados.
Esta seção contém os tópicos a seguir:
Como consultar arquivos com um atributo ou nome especificado (C#)
Mostra como pesquisar arquivos, examinando uma ou mais propriedades de seu objeto FileInfo.
Como agrupar arquivos por extensão (LINQ ) (C#)
Mostra como retornar grupos de objetos FileInfo com base em sua extensão de nome de arquivo.
Como consultar o número total de bytes em um conjunto de pastas (LINQ ) (C#)
Mostra como retornar o número total de bytes em todos os arquivos de uma árvore de diretórios especificada.
Como comparar o conteúdo de duas pastas (LINQ ) (C#)s
Mostra como retornar todos os arquivos que estão presentes em duas pastas especificadas e também todos os
arquivos que estão presentes em uma pasta, mas não na outra.
Como consultar o maior arquivo ou arquivos em uma árvore de diretório (LINQ ) (C#)
Mostra como retornar o maior ou o menor arquivo ou um número especificado de arquivos, em uma árvore de
diretório.
Como consultar arquivos duplicados em uma árvore de diretório (LINQ ) (C#)
Mostra como agrupar todos os nomes de arquivo que ocorrem em mais de um local em uma árvore de
diretórios especificada. Também mostra como realizar comparações mais complexas com base em um
comparador personalizado.
Como consultar o conteúdo de arquivos em uma pasta (LINQ ) (C#)
Mostra como iterar pelas pastas em uma árvore, abrir cada arquivo e consultar o conteúdo do arquivo.
Comentários
Há certa complexidade envolvida na criação de uma fonte de dados que representa o conteúdo do sistema de
arquivos com precisão e trata exceções de maneira elegante. Os exemplos nesta seção criam uma coleção de
instantâneos de objetos FileInfo que representa todos os arquivos em uma pasta raiz especificada e todas as
suas subpastas. O estado real de cada FileInfo pode ser alterado no tempo entre o momento em que você
começa e termina a execução de uma consulta. Por exemplo, você pode criar uma lista de objetos FileInfo para
usar como uma fonte de dados. Se você tentar acessar a propriedade Length em uma consulta, o objeto
FileInfo tentará acessar o sistema de arquivos para atualizar o valor de Length . Se o arquivo não existir, você
obterá uma FileNotFoundException em sua consulta, embora não esteja consultando diretamente o sistema de
arquivos. Algumas consultas nesta seção usam um método separado que consome essas exceções específicas
em determinados casos. Outra opção é manter a fonte de dados atualizada dinamicamente usando o
FileSystemWatcher.
Consulte também
LINQ to Objects (C#)
Como: Consultar arquivos com um atributo ou um
nome especificado (C#)
23/10/2019 • 2 minutes to read • Edit Online
Este exemplo mostra como localizar todos os arquivos que têm uma extensão de nome de arquivo especificada
(por exemplo ".txt") em uma árvore de diretório especificada. Ele também mostra como retornar tanto os arquivos
mais recentes como os mais antigo na árvore com base na hora de criação.
Exemplo
class FindFileByExtension
{
// This query will produce the full path for all .txt files
// under the specified folder including subfolders.
// It orders the list according to the file name.
static void Main()
{
string startFolder = @"c:\program files\Microsoft Visual Studio 9.0\";
Consulte também
LINQ to Objects (C#)
LINQ e diretórios de arquivos (C#)
Como: Agrupar arquivos por extensão (LINQ) (C#)
23/10/2019 • 3 minutes to read • Edit Online
Este exemplo mostra como o LINQ pode ser usado para realizar operações avançadas de classificação e
agrupamento em listas de arquivos ou pastas. Ele também mostra como paginar a saída na janela do console
usando os métodos Skip e Take.
Exemplo
A consulta a seguir mostra como agrupar o conteúdo de uma árvore de diretórios especificada, pela extensão de
nome de arquivo.
class GroupByExtension
{
// This query will sort all the files under the specified folder
// and subfolder into groups keyed by the file extension.
private static void Main()
{
// Take a snapshot of the file system.
string startFolder = @"c:\program files\Microsoft Visual Studio 9.0\Common7";
// This method specifically handles group queries of FileInfo objects with string keys.
// It can be modified to work for any long listings of data. Note that explicit typing
// must be used in method signatures. The groupbyExtList parameter is a query that produces
// groups of FileInfo objects with string keys.
private static void PageOutput(int rootLength,
IEnumerable<System.Linq.IGrouping<string, System.IO.FileInfo>>
groupByExtList)
{
// Flag to break out of paging loop.
bool goAgain = true;
// "3" = 1 line for extension + 1 for "Press any key" + 1 for input cursor.
int numLines = Console.WindowHeight - 3;
// Output only as many lines of the current group as will fit in the window.
do
{
Console.Clear();
Console.WriteLine(filegroup.Key == String.Empty ? "[none]" : filegroup.Key);
if (goAgain == false)
break;
}
}
}
A saída desse programa pode ser longa, dependendo dos detalhes do sistema de arquivos local e o que está
definido em startFolder . Para habilitar a exibição de todos os resultados, este exemplo mostra como paginá-los.
As mesmas técnicas podem ser aplicadas a aplicativos do Windows e aplicativos Web. Observe que, como o
código dispõe os itens em um grupo, é necessário um loop foreach aninhado. Há também alguma lógica
adicional para calcular a posição atual na lista e para permitir que o usuário interrompa a paginação e saia do
programa. Nesse caso específico, a consulta de paginação é executada nos resultados da consulta original
armazenados em cache. Em outros contextos, como LINQ to SQL, esse cache não é necessário.
Compilando o código
Criar um projeto de aplicativo de console em C# com diretivas using para os namespaces System.Linq e
System.IO.
Consulte também
LINQ to Objects (C#)
LINQ e diretórios de arquivos (C#)
Como: Consultar o número total de bytes em um
conjunto de pastas (LINQ) (C#)
23/10/2019 • 3 minutes to read • Edit Online
Este exemplo mostra como recuperar o número total de bytes usado por todos os arquivos em uma pasta
especificada e todas as suas subpastas.
Exemplo
O método Sum adiciona os valores de todos os itens selecionados na cláusula select . Você pode modificar essa
consulta para recuperar o maior ou o menor arquivo na árvore de diretório especificada chamando o método Min
ou Max em vez de Sum.
class QuerySize
{
public static void Main()
{
string startFolder = @"c:\program files\Microsoft Visual Studio 9.0\VC#";
// Return the total number of bytes in all the files under the specified folder.
long totalBytes = fileLengths.Sum();
Se precisar apenas contar o número de bytes em uma árvore de diretório especificada, você pode fazer isso com
mais eficiência sem criar uma consulta LINQ, que gera a sobrecarga de criação da coleção de lista como uma fonte
de dados. A utilidade da abordagem da LINQ aumenta conforme a consulta se torna mais complexa ou quando
você precisa executar várias consultas na mesma fonte de dados.
A consulta chama um método separado para obter o tamanho de arquivo. Ela faz isso para consumir a possível
exceção que será gerada se o arquivo tiver sido excluído em outro thread após o objeto FileInfo ter sido criado na
chamada para GetFiles . Embora o objeto FileInfo já tenha sido criado, a exceção poderá ocorrer porque um
objeto FileInfo tentará atualizar sua propriedade Length com o tamanho mais atual na primeira vez que a
propriedade foi acessada. Ao colocar essa operação em um bloco try-catch fora da consulta, o código segue a
regra de evitar operações em consultas que podem causar efeitos colaterais. Em geral, deve-se ter muito cuidado
ao consumir exceções para garantir que um aplicativo não seja deixado em um estado desconhecido.
Compilando o código
Criar um projeto de aplicativo de console em C# com diretivas using para os namespaces System.Linq e
System.IO.
Consulte também
LINQ to Objects (C#)
LINQ e diretórios de arquivos (C#)
Como comparar o conteúdo de duas pastas (LINQ)
(C#)
25/11/2019 • 3 minutes to read • Edit Online
NOTE
As técnicas mostradas aqui podem ser adaptadas para comparar sequências de objetos de qualquer tipo.
A classe FileComparer mostrada aqui demonstra como usar uma classe de comparação personalizada junto com
operadores de consulta padrão. A classe não se destina ao uso em cenários do mundo real. Ela apenas utiliza o
nome e o comprimento em bytes de cada arquivo para determinar se o conteúdo de cada pasta é idêntico ou não.
Em um cenário do mundo real, você deve modificar esse comparador para executar uma verificação mais rigorosa
de igualdade.
Exemplo
namespace QueryCompareTwoDirs
{
class CompareDirs
{
if (queryCommonFiles.Count() > 0)
{
Console.WriteLine("The following files are in both folders:");
foreach (var v in queryCommonFiles)
{
Console.WriteLine(v.FullName); //shows which items end up in result list
}
}
else
{
Console.WriteLine("There are no common files in the two folders.");
}
Consulte também
LINQ to Objects (C#)
LINQ e diretórios de arquivos (C#)
Como: Consultar o maior arquivo ou arquivos em
uma árvore de diretório (LINQ) (C#)
23/10/2019 • 5 minutes to read • Edit Online
Exemplo
O exemplo a seguir contém cinco consultas separadas que mostram como consultar e agrupar arquivos,
dependendo do tamanho do arquivo em bytes. Você pode modificar facilmente esses exemplos para basear a
consulta em outra propriedade do objeto FileInfo.
class QueryBySize
{
static void Main(string[] args)
{
QueryFilesBySize();
Console.WriteLine("Press any key to exit");
Console.ReadKey();
}
Console.WriteLine("The largest file under {0} is {1} with a length of {2} bytes",
startFolder, longestFile.FullName, longestFile.Length);
Console.WriteLine("The smallest file under {0} is {1} with a length of {2} bytes",
startFolder, smallestFile.FullName, smallestFile.Length);
Para retornar um ou mais objetos FileInfo completos, a consulta deve primeiro examinar cada um dos objetos na
fonte de dados e, em seguida, classificá-los segundo o valor de sua propriedade Length. Em seguida, ela pode
retornar um único elemento ou a sequência com os maiores tamanhos. Use First para retornar o primeiro
elemento em uma lista. Use Take para retornar o primeiro número n de elementos. Especifique uma ordem de
classificação decrescente para colocar os menores elementos no início da lista.
A consulta chama um método separado para obter o tamanho do arquivo em bytes para consumir a exceção
possível que ocorrerá caso um arquivo tenha sido excluído em outro thread no período desde que o objeto
FileInfo foi criado na chamada para GetFiles . Embora o objeto FileInfo já tenha sido criado, a exceção poderá
ocorrer porque um objeto FileInfo tentará atualizar sua propriedade Length usando o tamanho mais atual em
bytes na primeira vez que a propriedade foi acessada. Ao colocar essa operação em um bloco try-catch fora da
consulta, nós seguimos a regra de evitar operações em consultas que podem causar efeitos colaterais. Em geral,
deve-se ter muito cuidado ao consumir exceções para garantir que um aplicativo não seja deixado em um estado
desconhecido.
Compilando o código
Criar um projeto de aplicativo de console em C# com diretivas using para os namespaces System.Linq e
System.IO.
Consulte também
LINQ to Objects (C#)
LINQ e diretórios de arquivos (C#)
Como: Consultar arquivos duplicados em uma árvore
de diretório (LINQ) (C#)
23/10/2019 • 4 minutes to read • Edit Online
Às vezes, arquivos que têm o mesmo nome podem ser localizados em mais de uma pasta. Por exemplo, sob a
pasta de instalação do Visual Studio, várias pastas têm um arquivo readme.htm. Este exemplo mostra como
consultar esses nomes de arquivos duplicados sob uma pasta raiz especificada. O segundo exemplo mostra como
consultar arquivos cujo tamanho e LastWrite vezes também correspondem.
Exemplo
class QueryDuplicateFileNames
{
static void Main(string[] args)
{
// Uncomment QueryDuplicates2 to run that query.
QueryDuplicates();
// QueryDuplicates2();
int i = queryDupFiles.Count();
PageOutput<PortableKey, string>(queryDupFiles);
}
// "3" = 1 line for extension + 1 for "Press any key" + 1 for input cursor.
int numLines = Console.WindowHeight - 3;
// Output only as many lines of the current group as will fit in the window.
do
{
Console.Clear();
Console.WriteLine("Filename = {0}", filegroup.Key.ToString() == String.Empty ? "[none]" :
filegroup.Key.ToString());
if (goAgain == false)
break;
}
}
}
A primeira consulta usa uma chave simples para determinar uma correspondência. Ela localiza arquivos que têm o
mesmo nome, mas cujo conteúdo pode ser diferente. A segunda consulta usa uma chave composta para comparar
em relação a três propriedades do objeto FileInfo. É muito mais provável que essa consulta localize arquivos que
têm o mesmo nome e conteúdo semelhante ou idêntico.
Compilando o código
Criar um projeto de aplicativo de console em C# com diretivas using para os namespaces System.Linq e
System.IO.
Consulte também
LINQ to Objects (C#)
LINQ e diretórios de arquivos (C#)
Como consultar o conteúdo de arquivos de texto em
uma pasta (LINQ) (C#)
25/11/2019 • 2 minutes to read • Edit Online
Este exemplo mostra como consultar todos os arquivos em uma árvore de diretório especificada, abrir cada
arquivo e inspecionar seu conteúdo. Este tipo de técnica pode ser usado para criar índices ou inverter os índices do
conteúdo de uma árvore de diretório. Uma pesquisa de cadeia de caracteres simples é executada neste exemplo.
No entanto, os tipos de correspondência de padrões mais complexos podem ser executados com uma expressão
regular. Para obter mais informações, consulte como combinar consultas LINQ com expressões regulares (C#).
Exemplo
class QueryContents
{
public static void Main()
{
// Modify this path as necessary.
string startFolder = @"c:\program files\Microsoft Visual Studio 9.0\";
Compilando o código
Criar um projeto de aplicativo de console em C# com diretivas using para os namespaces System.Linq e
System.IO.
Consulte também
LINQ e diretórios de arquivos (C#)
LINQ to Objects (C#)
Como consultar um ArrayList com LINQ (C#)
25/11/2019 • 2 minutes to read • Edit Online
Ao usar a LINQ para consultar coleções IEnumerable não genéricas como ArrayList, você deve declarar
explicitamente o tipo da variável de intervalo para refletir o tipo específico dos objetos na coleção. Por exemplo,
se você tiver um ArrayList de objetos Student , sua cláusula from deverá ter uma aparência semelhante a esta:
Especificando o tipo da variável de intervalo, você está convertendo cada item na ArrayList em um Student .
O uso de uma variável de intervalo de tipo explícito em uma expressão de consulta é equivalente a chamar o
método Cast. Cast lança uma exceção se a conversão especificada não puder ser realizada. Cast e OfType são os
dois métodos de operador de consulta padrão que operam em tipos IEnumerable não genéricos. Para obter
mais informações, consulte Relacionamentos de tipo em operações de consulta LINQ.
Exemplo
O exemplo a seguir mostra uma consulta simples sobre um ArrayList. Observe que este exemplo usa os
inicializadores de objeto quando o código chama o método Add, mas isso não é um requisito.
using System;
using System.Collections;
using System.Linq;
namespace NonGenericLINQ
{
public class Student
{
public string FirstName { get; set; }
public string LastName { get; set; }
public int[] Scores { get; set; }
}
class Program
{
static void Main(string[] args)
{
ArrayList arrList = new ArrayList();
arrList.Add(
new Student
{
FirstName = "Svetlana", LastName = "Omelchenko", Scores = new int[] { 98, 92, 81, 60
}
});
arrList.Add(
new Student
{
FirstName = "Claire", LastName = "O’Donnell", Scores = new int[] { 75, 84, 91, 39 }
});
arrList.Add(
new Student
{
FirstName = "Sven", LastName = "Mortensen", Scores = new int[] { 88, 94, 65, 91 }
});
arrList.Add(
new Student
{
FirstName = "Cesar", LastName = "Garcia", Scores = new int[] { 97, 89, 85, 82 }
});
Consulte também
LINQ to Objects (C#)
Como adicionar métodos personalizados para
consultas LINQ (C#)
25/11/2019 • 8 minutes to read • Edit Online
Você pode estender o conjunto de métodos que você pode usar para consultas LINQ, adicionando os métodos de
extensão à interface IEnumerable<T>. Por exemplo, além do padrão médio ou máximo de operações, você pode
criar um método de agregação personalizado para calcular um único valor de uma sequência de valores. Você
também pode criar um método que funciona como um filtro personalizado ou como uma transformação de dados
específicos para uma sequência de valores e que retorna uma nova sequência. Exemplos desses métodos são
Distinct, Skip e Reverse.
Ao estender a interface IEnumerable<T>, você pode aplicar seus métodos personalizados para qualquer coleção
enumerável. Para obter mais informações, consulte Métodos de extensão.
if (sortedList.Count() % 2 == 0)
{
// Even number of items.
return (sortedList.ElementAt(itemIndex) + sortedList.ElementAt(itemIndex - 1)) / 2;
}
else
{
// Odd number of items.
return sortedList.ElementAt(itemIndex);
}
}
}
Você chama esse método de extensão para qualquer coleção enumerável da mesma maneira que chama outros
métodos de agregação da interface IEnumerable<T>.
O exemplo de código a seguir mostra como usar o método Median para uma matriz do tipo double .
double[] numbers1 = { 1.9, 2, 8, 4, 5.7, 6, 7.2, 0 };
/*
This code produces the following output:
//int overload
Agora você pode chamar as sobrecargas Median para os tipos integer e double , conforme mostrado no código
a seguir:
int[] numbers2 = { 1, 2, 3, 4, 5 };
/*
This code produces the following output:
// Generic overload.
Agora você pode chamar o método Median para uma sequência de objetos de qualquer tipo. Se o tipo não tem
sua própria sobrecarga de método, você precisa passar um parâmetro delegado. No C# você pode usar uma
expressão lambda para essa finalidade. Além disso, no Visual Basic, se você usar a cláusula Aggregate ou
Group By em vez da chamada de método, você pode passar qualquer valor ou expressão que estiver no escopo
dessa cláusula.
O exemplo de código a seguir mostra como chamar o método Median para uma matriz de inteiros e para uma
matriz de cadeias de caracteres. Será calculada a mediana dos comprimentos das cadeias de caracteres na matriz.
O exemplo também mostra como passar o parâmetro delegado Func<T,TResult> ao método Median para cada
caso.
int[] numbers3 = { 1, 2, 3, 4, 5 };
/*
You can use the num=>num lambda expression as a parameter for the Median method
so that the compiler will implicitly convert its value to double.
If there is no implicit conversion, the compiler will display an error message.
*/
// With the generic overload, you can also use numeric properties of objects.
/*
This code produces the following output:
Integer: Median = 3
String: Median = 4
*/
int i = 0;
i++;
}
return list;
}
Você pode chamar esse método de extensão para qualquer coleção enumerável exatamente como chamaria
outros métodos da interface IEnumerable<T>, conforme mostrado no código a seguir:
a
c
e
*/
Consulte também
IEnumerable<T>
Métodos de Extensão
Visão geral do LINQ to XML (C#)
23/10/2019 • 8 minutes to read • Edit Online
O LINQ to XML fornece uma interface de programação XML na memória que aproveita a .NET LINQ
(Consulta Integrada à Linguagem) Framework. O LINQ to XML usa os recursos do .NET e pode ser
comparado a uma interface de programação XML de Modelo de Objeto do Documento (DOM ) atualizada e
reprojetada.
O XML tem sido amplamente adotado como um modo de formatar dados em muitos contextos. Por exemplo,
você pode encontrar XML na Web, em arquivos de configuração, em arquivos do Microsoft Office Word e em
bancos de dados.
O LINQ to XML é uma abordagem atualizada e redesenhada para a programação com XML. Ele oferece os
recursos de modificação de documentos na memória do DOM (Document Object Model), bem como o
suporte a expressões de consulta LINQ. Embora essas expressões de consulta sejam sintaticamente diferentes
de XPath, elas oferecem uma funcionalidade semelhante.
// Load the XML file from our project directory containing the purchase orders
var filename = "PurchaseOrder.xml";
var currentDirectory = Directory.GetCurrentDirectory();
var purchaseOrderFilepath = Path.Combine(currentDirectory, filename);
Como outro exemplo, você poderia querer uma lista, classificada por número de peça, dos itens com um valor
maior que $ 100. Para obter essas informações, você poderia executar a seguinte consulta:
// Load the XML file from our project directory containing the purchase orders
var filename = "PurchaseOrder.xml";
var currentDirectory = Directory.GetCurrentDirectory();
var purchaseOrderFilepath = Path.Combine(currentDirectory, filename);
Além desses recursos de LINQ, o LINQ to XML oferece uma interface de programação XML aprimorada.
Usando o LINQ to XML, você pode:
Carregar XML de arquivos ou fluxos.
Serializar o XML em arquivos ou fluxos.
Crie XML a partir do zero usando a construção funcional.
Consultar o XML usando eixos do tipo XPath.
Manipular a árvore XML na memória usando métodos como Add, Remove, ReplaceWith e SetValue.
Validar árvores XML usando XSD.
Usar uma combinação desses recursos para transformar árvores XML de uma forma em outra.
Criando árvores XML
Uma das vantagens mais significativas da programação com LINQ to XML é que é fácil criar árvores XML. Por
exemplo, para criar uma árvore XML pequena, você pode escrever código da seguinte forma:
XElement contacts =
new XElement("Contacts",
new XElement("Contact",
new XElement("Name", "Patrick Hines"),
new XElement("Phone", "206-555-0144",
new XAttribute("Type", "Home")),
new XElement("phone", "425-555-0145",
new XAttribute("Type", "Work")),
new XElement("Address",
new XElement("Street1", "123 Main St"),
new XElement("City", "Mercer Island"),
new XElement("State", "WA"),
new XElement("Postal", "68042")
)
)
);
Consulte também
Referência (LINQ to XML )
LINQ to XML versus DOM (C#)
LINQ to XML versus Outras Tecnologias XML
System.Xml.Linq
LINQ to XML e DOM (C#)
23/10/2019 • 10 minutes to read • Edit Online
Esta seção descreve algumas das principais diferenças entre o LINQ to XML e a atual API de programação de
XML predominante, o W3C DOM (Modelo de Objeto do Documento).
Este estilo de codificação não fornece visualmente muitas informações sobre a estrutura da árvore XML. O LINQ
to XML dá suporte a esta abordagem para construir uma árvore XML, mas também dá suporte a uma abordagem
alternativa, a construção funcional. A construção funcional usa os construtores XElement e de XAttribute para criar
uma árvore XML.
Veja como você construiria a mesma árvore XML usando a construção funcional LINQ to XML:
XElement contacts =
new XElement("Contacts",
new XElement("Contact",
new XElement("Name", "Patrick Hines"),
new XElement("Phone", "206-555-0144",
new XAttribute("Type", "Home")),
new XElement("phone", "425-555-0145",
new XAttribute("Type", "Work")),
new XElement("Address",
new XElement("Street1", "123 Main St"),
new XElement("City", "Mercer Island"),
new XElement("State", "WA"),
new XElement("Postal", "68042")
)
)
);
Observe que o recuo do código para construir a árvore XML mostra a estrutura do XML subjacente.
Para obter mais informações, consulte Criando árvores XML (C#) .
Compare isso para o W3C DOM, no qual o documento XML é usado como um contêiner lógico para a árvore
XML. No DOM, os nós XML, incluindo elementos e atributos, devem ser criados no contexto de um documento
XML. Aqui está um fragmento de código para criar um elemento de nome DOM:
Se você quiser usar um elemento em vários documentos, deverá importar os nós nos documentos. LINQ to
XML evita essa camada de complexidade.
Ao usar LINQ to XML, você usará a classe XDocument somente se você desejar adicionar um comentário ou uma
instrução de processamento no nível raiz do documento.
Consulte também
Introdução (LINQ to XML )
LINQ to XML e outras tecnologias XML
23/10/2019 • 7 minutes to read • Edit Online
Este tópico compara o LINQ to XML às seguintes tecnologias XML: XmlReader, XSLT, MSXML e XmlLite. Estas
informações podem ajudá-lo a decidir qual tecnologia usar.
Para ver uma comparação do LINQ to XML com DOM (Modelo de Objeto do Documento), consulte LINQ to
XML vs. DOM (C#).
Consulte também
Introdução (LINQ to XML )
Programação funcional versus procedural (LINQ to
XML) (C#)
23/10/2019 • 3 minutes to read • Edit Online
Consulte também
Visão geral da programação LINQ to XML (C#)
Visão geral de classes LINQ to XML (C#)
23/10/2019 • 5 minutes to read • Edit Online
Este tópico fornece uma lista das classes LINQ to XML no namespace System.Xml.Linq e uma breve descrição de
cada.
Consulte também
Visão geral da programação LINQ to XML (C#)
Visão geral da classe XElement (C#)
23/10/2019 • 4 minutes to read • Edit Online
A classe XElement é uma das classes fundamentais no LINQ to XML. Representa um elemento XML. Você pode
usar essa classe para criar elementos; alterar o conteúdo do elemento; adicionar, alterar ou excluir elementos filho;
adicionar atributos a um elemento; ou serializar o conteúdo de um elemento no formulário de texto. Você também
pode interoperar com outras classes no System.Xml, como XmlReader, XmlWriter e XslCompiledTransform.
Este tópico descreve a funcionalidade fornecida pela classe XElement.
XElement contacts =
new XElement("Contacts",
new XElement("Contact",
new XElement("Name", "Patrick Hines"),
new XElement("Phone", "206-555-0144"),
new XElement("Address",
new XElement("Street1", "123 Main St"),
new XElement("City", "Mercer Island"),
new XElement("State", "WA"),
new XElement("Postal", "68042")
)
)
);
Outra técnica muito comum para criar uma árvore XML envolve o uso dos resultados de uma consulta LINQ para
popular uma árvore XML, conforme mostrado no exemplo o seguir:
XElement srcTree = new XElement("Root",
new XElement("Element", 1),
new XElement("Element", 2),
new XElement("Element", 3),
new XElement("Element", 4),
new XElement("Element", 5)
);
XElement xmlTree = new XElement("Root",
new XElement("Child", 1),
new XElement("Child", 2),
from el in srcTree.Elements()
where (int)el > 2
select el
);
Console.WriteLine(xmlTree);
<Root>
<Child>1</Child>
<Child>2</Child>
<Element>3</Element>
<Element>4</Element>
<Element>5</Element>
</Root>
Consulte também
Visão geral da programação LINQ to XML (C#)
Visão geral da classe XAttribute (C#)
23/10/2019 • 3 minutes to read • Edit Online
Os atributos são pares nome/valor que são associados a um elemento. A classe de XAttribute representa atributos
XML.
Visão geral
Trabalhar com atributos em LINQ to XML é semelhante a trabalhar com elementos. Os construtores são
semelhantes. Os métodos que você usa para recuperar coleções deless são semelhantes. Uma expressão de
consulta LINQ para uma coleção de atributos se parece muito com uma expressão de consulta LINQ para uma
coleção de elementos.
A ordem em que os atributos foram adicionados a um elemento é preservada. Isto é, quando você itera através de
atributos, você ver na mesma ordem que foram adicionados.
O construtor de XAttribute
O seguinte construtor de classe XAttribute é aquele que você usará mais comumente:
CONSTRUTOR DESCRIÇÃO
<Phone Type="Home">555-555-5555</Phone>
<Customers>
<Customer>
<Name>John Doe</Name>
<PhoneNumbers>
<Phone type="home">555-555-5555</Phone>
<Phone type="work">666-666-6666</Phone>
</PhoneNumbers>
</Customer>
</Customers>
Consulte também
Visão geral da programação LINQ to XML (C#)
Visão geral da classe XDocument (C#)
23/10/2019 • 3 minutes to read • Edit Online
Componentes de XDocument
Um XDocument pode conter os seguintes elementos:
Um objeto XDeclaration. XDeclaration permite que você especifique as partes pertinentes de uma
declaração XML: a versão XML, a codificação do documento e se o documento XML é autônomo.
Um objeto XElement. Esse é o nó raiz do documento XML.
Qualquer número de objetos XProcessingInstruction. Uma instrução de processamento comunica
informações a um aplicativo que processa o XML.
Qualquer número de objetos XComment. Os comentários serão irmãos do elemento raiz. O objeto
XComment não pode ser o primeiro argumento da lista, pois não é válido um documento XML iniciar com
um comentário.
Um XDocumentType para o DTD.
Quando você serializa um XDocument, mesmo que XDocument.Declaration seja null , a saída terá uma
declaração XML se o gravador tiver Writer.Settings.OmitXmlDeclaration definido como false (o padrão).
Por padrão, o LINQ to XML define a versão como “1.0 " e a codificação como “utf-8”.
Usando XDocument
Para construir um XDocument, use a construção funcional exatamente como você faz para criar objetos XElement.
O código a seguir cria um objeto XDocument e seus objetos independentes associados.
XDocument d = new XDocument(
new XComment("This is a comment."),
new XProcessingInstruction("xml-stylesheet",
"href='mystyle.css' title='Compact' type='text/css'"),
new XElement("Pubs",
new XElement("Book",
new XElement("Title", "Artifacts of Roman Civilization"),
new XElement("Author", "Moreno, Jordao")
),
new XElement("Book",
new XElement("Title", "Midieval Tools and Implements"),
new XElement("Author", "Gazit, Inbar")
)
),
new XComment("This is another comment.")
);
d.Declaration = new XDeclaration("1.0", "utf-8", "true");
Console.WriteLine(d);
d.Save("test.xml");
Consulte também
Visão geral da programação LINQ to XML (C#)
Como criar LINQ to XML exemplos (C#)
25/11/2019 • 2 minutes to read • Edit Online
Os vários snippets e exemplos nesta documentação usam classes e tipos de uma variedade de namespaces. Para
compilar o código em C#, você precisará fornecer diretivas apropriadas de using .
Exemplo
O código a seguir contém diretivas de using que os exemplos de C# exigem para compilar e executar. Nem todas
as diretivas de using são necessárias para todos os exemplos.
using System;
using System.Diagnostics;
using System.Collections;
using System.Collections.Generic;
using System.Collections.Specialized;
using System.Text;
using System.Linq;
using System.Xml;
using System.Xml.Linq;
using System.Xml.Schema;
using System.Xml.XPath;
using System.Xml.Xsl;
using System.IO;
using System.Threading;
using System.Reflection;
using System.IO.Packaging;
Consulte também
Visão geral da programação LINQ to XML (C#)
Construção funcional (LINQ to XML) (C#)
23/10/2019 • 2 minutes to read • Edit Online
LINQ to XML fornece uma maneira eficiente de criar elementos XML chamada construção funcional. Construção
funcional é a capacidade de criar uma árvore XML em uma única instrução.
Há vários recursos chave da interface de programação do LINQ to XML que permitem a construção funcional:
O construtor XElement utiliza vários tipos de argumentos para o conteúdo. Por exemplo, você pode passar
outro objeto XElement, que se torna um elemento filho. Você pode passar um objeto XAttribute, que se
torna um atributo do elemento. Ou você pode passar qualquer outro tipo de objeto, que é convertido em
uma cadeia de caracteres e torna-se o conteúdo de texto do elemento.
O construtor XElement utiliza uma matriz de params do tipo Object, para que você possa passar qualquer
número de objetos para o construtor. Isso permite que você crie um elemento que tem o conteúdo
complexo.
Se um objeto implementar IEnumerable<T>, a coleção no objeto será enumerada, e todos os itens da
coleção serão adicionados. Se a coleção contiver objetos XElement ou XAttribute, cada item da coleção será
adicionado separadamente. Isso é importante porque permite que você passe os resultados de uma
consulta LINQ para o construtor.
Esses recursos permitem escrever código para criar uma árvore XML. A seguir está um exemplo:
XElement contacts =
new XElement("Contacts",
new XElement("Contact",
new XElement("Name", "Patrick Hines"),
new XElement("Phone", "206-555-0144"),
new XElement("Address",
new XElement("Street1", "123 Main St"),
new XElement("City", "Mercer Island"),
new XElement("State", "WA"),
new XElement("Postal", "68042")
)
)
);
Esses recursos também permitem que você escreva código que usa os resultados de consultas LINQ quando cria
uma árvore XML, da seguinte maneira:
XElement srcTree = new XElement("Root",
new XElement("Element", 1),
new XElement("Element", 2),
new XElement("Element", 3),
new XElement("Element", 4),
new XElement("Element", 5)
);
XElement xmlTree = new XElement("Root",
new XElement("Child", 1),
new XElement("Child", 2),
from el in srcTree.Elements()
where (int)el > 2
select el
);
Console.WriteLine(xmlTree);
<Root>
<Child>1</Child>
<Child>2</Child>
<Element>3</Element>
<Element>4</Element>
<Element>5</Element>
</Root>
Criando árvores XML em C# (LINQ to XML)
23/10/2019 • 8 minutes to read • Edit Online
Construindo elementos
As assinaturas dos construtores XElement e XAttribute permitem que você passe o conteúdo do elemento ou
atributo como argumentos para o construtor. Como um dos construtores recebe um número variável de
argumentos, você pode passar qualquer número de elementos filhos. Naturalmente, cada um desses elementos
filhos pode conter seus próprios elementos filhos. Para qualquer elemento, você pode adicionar a quantidade
desejada de atributos.
Ao adicionar objetos XNode (inclusive XElement) ou XAttribute, se o novo conteúdo não tiver um pai, os objetos
serão simplesmente anexados à árvore XML. Se o novo conteúdo já tiver um pai e fizer parte de outra árvore
XML, o novo conteúdo será clonado, e o conteúdo recém-clonado será anexado à árvore XML. O último exemplo
deste tópico demonstra isso.
Para criar um contacts XElement, você pode usar o seguinte código:
XElement contacts =
new XElement("Contacts",
new XElement("Contact",
new XElement("Name", "Patrick Hines"),
new XElement("Phone", "206-555-0144"),
new XElement("Address",
new XElement("Street1", "123 Main St"),
new XElement("City", "Mercer Island"),
new XElement("State", "WA"),
new XElement("Postal", "68042")
)
)
);
Se recuado corretamente, o código para criar objetos XElement é muito parecido com a estrutura do XML
subjacente.
Construtores de XElement
A classe XElement usa os seguintes construtores na construção funcional. Observe que há outros construtores
para XElement, mas como eles não são usados na construção funcional, eles não são listados aqui.
CONSTRUTOR DESCRIÇÃO
XElement(XName name, object content) Cria um XElement. O parâmetro name especifica o nome do
elemento; content especifica o conteúdo do elemento.
XElement(XName name) Cria um XElement com seu XName inicializado para o nome
especificado.
CONSTRUTOR DESCRIÇÃO
XElement(XName name, params object[] content) Cria um XElement com seu XName inicializado para o nome
especificado. Os atributos e/ou elementos filhos são criados a
partir do conteúdo da lista de parâmetros.
O parâmetro content é muito flexível. Ele dá suporte a qualquer tipo de objeto que seja um filho válido de um
XElement. As seguintes regras se aplicam aos diferentes tipos de objetos passados neste parâmetro:
Uma cadeia de caracteres é adicionada como conteúdo de texto.
Um XElement é adicionado como um elemento filho.
Um XAttribute é adicionado como um atributo.
Um XProcessingInstruction, XComment ou XText é adicionado como conteúdo filho.
Um IEnumerable é enumerado, e essas regras são aplicadas recursivamente aos resultados.
Para qualquer outro tipo, seu método ToString é chamado e o resultado é adicionado como conteúdo de
texto.
Criando um XElement com conteúdo
Você pode criar um XElement contendo o conteúdo simples com uma única chamada de método. Para fazer isso,
especifique o conteúdo como o segundo parâmetro, como segue:
<Customer>Adventure Works</Customer>
Você pode passar qualquer tipo de objeto como o conteúdo. Por exemplo, o seguinte código cria um elemento
que contém um número de ponto flutuante como conteúdo:
<Cost>324.5</Cost>
O número de ponto flutuante foi convertido e passado para o construtor. O número é convertido em uma cadeia
de caracteres e usado como o conteúdo do elemento.
Criando um XElement com um elemento filho
Se você passar uma instância da classe XElement para o argumento de conteúdo, o construtor criará um elemento
com um elemento filho:
<ShippingUnit>
<Cost>324.5</Cost>
</ShippingUnit>
<Address>
<Street1>123 Main St</Street1>
<City>Mercer Island</City>
<State>WA</State>
<Postal>68042</Postal>
</Address>
Por extensão ao exemplo anterior, você pode criar uma árvore XML inteira, desta forma:
XElement contacts =
new XElement("Contacts",
new XElement("Contact",
new XElement("Name", "Patrick Hines"),
new XElement("Phone", "206-555-0144"),
new XElement("Address",
new XElement("Street1", "123 Main St"),
new XElement("City", "Mercer Island"),
new XElement("State", "WA"),
new XElement("Postal", "68042")
)
)
);
Console.WriteLine(contacts);
<Contacts>
<Contact>
<Name>Patrick Hines</Name>
<Phone>206-555-0144</Phone>
<Address>
<Street1>123 Main St</Street1>
<City>Mercer Island</City>
<State>WA</State>
<Postal>68042</Postal>
</Address>
</Contact>
</Contacts>
Criando um XElement com um XAttribute
Se você passar uma instância da classe XAttribute para o argumento de conteúdo, o construtor criará um
elemento com um atributo:
<Phone Type="Home">555-555-5555</Phone>
<Customer />
Consulte também
Criando árvores XML (C#)
Como analisar uma cadeia de caracteres (C#)
25/11/2019 • 2 minutes to read • Edit Online
Este tópico mostra como analisar uma cadeia de caracteres para criar uma árvore XML no C#.
Exemplo
O código C# a seguir mostra como analisar uma cadeia de caracteres XML:
O nó de Contacts raiz tem dois nós Contact . Para acessar alguns dados específicos em seu XML analisado, use o
método XElement. Elements () , que, nesse caso, retorna os elementos filho do nó de Contacts raiz. O exemplo a
seguir imprime o primeiro nó Contact no console:
Consulte também
Como localizar um elemento com um atributo específico (C#)
Como: Carregar o XML de um arquivo (C#)
23/10/2019 • 2 minutes to read • Edit Online
Este tópico mostra como carregar XML de um URI usando o método XElement.Load.
Exemplo
O seguinte exemplo mostra como carregar um documento XML de um arquivo. O seguinte exemplo carrega
books.xml e gera a árvore XML no console.
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: Livros (LINQ to XML ).
<Catalog>
<Book id="bk101">
<Author>Garghentini, Davide</Author>
<Title>XML Developer's Guide</Title>
<Genre>Computer</Genre>
<Price>44.95</Price>
<PublishDate>2000-10-01</PublishDate>
<Description>An in-depth look at creating applications
with XML.</Description>
</Book>
<Book id="bk102">
<Author>Garcia, Debra</Author>
<Title>Midnight Rain</Title>
<Genre>Fantasy</Genre>
<Price>5.95</Price>
<PublishDate>2000-12-16</PublishDate>
<Description>A former architect battles corporate zombies,
an evil sorceress, and her own childhood to become queen
of the world.</Description>
</Book>
</Catalog>
Preservar espaço em branco para carregar ou ao
analisar XML
23/10/2019 • 3 minutes to read • Edit Online
Este tópico descreve como controlar o comportamento de espaço em branco de LINQ to XML.
Um cenário comum é ler o XML recuado, criar uma árvore XML na memória sem nenhum nó de texto de espaço
em branco (isto é, não preservar espaço em branco), executar algumas operações no XML e, em seguida, salvar o
XML com recuo. Quando você serializa o XML com formatação, somente os espaços em branco significativos na
árvore XML são preservados. Este é o comportamento padrão para LINQ to XML.
Outro cenário comum é ler e modificar XML que já foi recuado intencionalmente. Você pode não querer modificar
este recuo de nenhuma forma. Para fazer isso em LINQ to XML, você preserva o espaço em branco quando você
carregar ou analisar XML e desativa o formatação quando você serializa XML.
Este tópico descreve o comportamento de espaço em branco dos métodos que preenchem árvores XML. Para
obter informações sobre o controle de espaço em branco ao serializar árvores XML, consulte Preservar espaço em
branco ao serializar.
Exemplo
O código a seguir tenta analisar XML válido:
try {
XElement contacts = XElement.Parse(
@"<Contacts>
<Contact>
<Name>Jim Wilson</Name>
</Contact>
</Contcts>");
Console.WriteLine(contacts);
}
catch (System.Xml.XmlException e)
{
Console.WriteLine(e.Message);
}
The 'Contacts' start tag on line 1 does not match the end tag of 'Contcts'. Line 5, position 13.
Para obter informações sobre as exceções que você pode esperar XElement.Parse, XDocument.Parse,
XElement.Load, e métodos de XDocument.Load lançar, consulte a documentação de XmlReader .
Como criar uma árvore a partir de um XmlReaderC#
()
25/11/2019 • 2 minutes to read • Edit Online
Este tópico mostra como criar uma árvore de XML diretamente de XmlReader. Para criar um XElement de um
XmlReader, você deverá posicionar o XmlReader em um nó de elemento. O XmlReader ignorará comentários e
instruções de processamento, mas se o XmlReader estiver posicionado em um nó de texto, um erro será gerado.
Para evitar esses erros, sempre posicione o XmlReader em um elemento antes de criar uma árvore de XML do
XmlReader.
Exemplo
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: livros (LINQ to XML ).
O código a seguir cria um objeto T:System.Xml.XmlReader e depois lê os nós até encontrar o primeiro nó do
elemento. Ele em seguida carrega o objeto XElement.
XmlReader r = XmlReader.Create("books.xml");
while (r.NodeType != XmlNodeType.Element)
r.Read();
XElement e = XElement.Load(r);
Console.WriteLine(e);
<Catalog>
<Book id="bk101">
<Author>Garghentini, Davide</Author>
<Title>XML Developer's Guide</Title>
<Genre>Computer</Genre>
<Price>44.95</Price>
<PublishDate>2000-10-01</PublishDate>
<Description>An in-depth look at creating applications
with XML.</Description>
</Book>
<Book id="bk102">
<Author>Garcia, Debra</Author>
<Title>Midnight Rain</Title>
<Genre>Fantasy</Genre>
<Price>5.95</Price>
<PublishDate>2000-12-16</PublishDate>
<Description>A former architect battles corporate zombies,
an evil sorceress, and her own childhood to become queen
of the world.</Description>
</Book>
</Catalog>
Consulte também
Analisando XML (C#)
Como: Transmitir fragmentos XML de um XmlReader
(C#)
23/10/2019 • 3 minutes to read • Edit Online
Quando você tem que processa grandes arquivos XML, talvez não seja possível carregar a árvore inteira XML na
memória. Este tópico mostra como passar informações usando XmlReader.
Um dos modos de efetivas usar XmlReader para ler objetos de XElement é escrever seu próprio método
personalizado do eixo. Um método do eixo normalmente retorna uma coleção como IEnumerable<T> de
XElement, conforme mostrado no exemplo neste tópico. No método personalizado do eixo, depois de criar o
fragmento XML chamando o método ReadFrom , retornar a coleção usando yield return . Isso fornece a
semântica de execução adiada ao método personalizado do eixo.
Quando você cria uma árvore XML de um objeto de XmlReader , XmlReader deve ser posicionado em um
elemento. O método de ReadFrom não retorna até que lê a marca do elemento.
Se você desejar criar uma árvore parcial, você pode criar uma instância XmlReader, posiciona o leitor no nó que
você deseja converter a XElement uma árvore e em seguida, cria o objeto de XElement .
O tópico Como: Transmitir fragmentos XML com acesso a informações de cabeçalho (C#) contém informações e
um exemplo de como transmitir um documento mais complexo.
O tópico Como: Executar a transformação de streaming de documentos XML grandes (C#) contém um exemplo de
como usar o LINQ to XML para transformar documentos XML muito grandes, mantendo um volume de memória
pequeno.
Exemplo
Este exemplo cria um método personalizado do eixo. Você pode consultá-lo usando uma consulta LINQ. O
método de eixo personalizado StreamRootChildDoc é um método que foi projetado especificamente para ler um
documento que tenha um elemento Child de repetição.
static IEnumerable<XElement> StreamRootChildDoc(StringReader stringReader)
{
using (XmlReader reader = XmlReader.Create(stringReader))
{
reader.MoveToContent();
// Parse the file and display each of the nodes.
while (reader.Read())
{
switch (reader.NodeType)
{
case XmlNodeType.Element:
if (reader.Name == "Child") {
XElement el = XElement.ReadFrom(reader) as XElement;
if (el != null)
yield return el;
}
break;
}
}
}
}
IEnumerable<string> grandChildData =
from el in StreamRootChildDoc(new StringReader(markup))
where (int)el.Attribute("Key") > 1
select (string)el.Element("GrandChild");
bbb
ccc
Nesse exemplo, o documento de origem é muito pequeno. No entanto, mesmo se houver milhões de elementos
de Child , este exemplo ainda terá uma pegada pequena de memória.
Como: Popular uma árvore XML com um XmlWriter
(LINQ to XML) (C#)
23/10/2019 • 2 minutes to read • Edit Online
Uma maneira de preencher uma árvore XML é usar CreateWriter para criar um XmlWriter e escrever no
XmlWriter. A árvore XML é preenchida com todos os nós que são escritos no XmlWriter.
Você normalmente usa este método quando usa LINQ to XML com outra classe que espera escrever em um
XmlWriter, por exemplo, XslCompiledTransform.
Exemplo
Um uso possível para CreateWriter é ao chamar uma transformação XSLT. Este exemplo cria uma árvore XML, cria
XmlReader da árvore XML, cria um novo documento e, em seguida, cria XmlWriter para gravar no novo
documento. Ele então chama a transformação XSLT, passando no XmlReader e no XmlWriter. Depois que a
transformação for concluída com êxito, a nova árvore XML será preenchida com os resultados da transformação.
Console.WriteLine(newTree);
Consulte também
CreateWriter
XmlWriter
XslCompiledTransform
Criando árvores XML (C#)
Como: Validar usando o XSD (LINQ to XML) (C#)
23/10/2019 • 2 minutes to read • Edit Online
O namespace System.Xml.Schema contém métodos de extensão que facilitam a validação de uma árvore XML em
um arquivo XSD. Para obter mais informações, consulte a documentação do método Validate.
Exemplo
O exemplo a seguir cria um XmlSchemaSet e, em seguida, valida dois objetos XDocument no conjunto de
esquemas. Um dos documentos é válido, o outro não.
string xsdMarkup =
@"<xsd:schema xmlns:xsd='http://www.w3.org/2001/XMLSchema'>
<xsd:element name='Root'>
<xsd:complexType>
<xsd:sequence>
<xsd:element name='Child1' minOccurs='1' maxOccurs='1'/>
<xsd:element name='Child2' minOccurs='1' maxOccurs='1'/>
</xsd:sequence>
</xsd:complexType>
</xsd:element>
</xsd:schema>";
XmlSchemaSet schemas = new XmlSchemaSet();
schemas.Add("", XmlReader.Create(new StringReader(xsdMarkup)));
Console.WriteLine("Validating doc1");
bool errors = false;
doc1.Validate(schemas, (o, e) =>
{
Console.WriteLine("{0}", e.Message);
errors = true;
});
Console.WriteLine("doc1 {0}", errors ? "did not validate" : "validated");
Console.WriteLine();
Console.WriteLine("Validating doc2");
errors = false;
doc2.Validate(schemas, (o, e) =>
{
Console.WriteLine("{0}", e.Message);
errors = true;
});
Console.WriteLine("doc2 {0}", errors ? "did not validate" : "validated");
Validating doc2
The element 'Root' has invalid child element 'Child3'. List of possible elements expected: 'Child2'.
doc2 did not validate
Exemplo
O seguinte exemplo valida que o documento XML de Arquivo XML de exemplo: Clientes e ordens (LINQ to XML )
é válido de acordo com o esquema de Arquivo XSD de exemplo: Clientes e ordens. Ele altera o documento XML de
origem. Ele altera o atributo CustomerID no primeiro cliente. Depois da alteração, os pedidos se referirão a um
cliente que não existe, portanto, o documento XML não será mais validado.
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: Clientes e ordens (LINQ to XML ).
Este exemplo usa o seguinte esquema XSD: Arquivo XSD de exemplo: Clientes e ordens.
Console.WriteLine("Attempting to validate");
XDocument custOrdDoc = XDocument.Load("CustomersOrders.xml");
bool errors = false;
custOrdDoc.Validate(schemas, (o, e) =>
{
Console.WriteLine("{0}", e.Message);
errors = true;
});
Console.WriteLine("custOrdDoc {0}", errors ? "did not validate" : "validated");
Console.WriteLine();
// Modify the source document so that it will not validate.
custOrdDoc.Root.Element("Orders").Element("Order").Element("CustomerID").Value = "AAAAA";
Console.WriteLine("Attempting to validate after modification");
errors = false;
custOrdDoc.Validate(schemas, (o, e) =>
{
Console.WriteLine("{0}", e.Message);
errors = true;
});
Console.WriteLine("custOrdDoc {0}", errors ? "did not validate" : "validated");
Attempting to validate
custOrdDoc validated
Consulte também
Validate
Criando árvores XML (C#)
Conteúdo válido de objetos XElement e XDocument
23/10/2019 • 4 minutes to read • Edit Online
Este tópico descreve os argumentos válidos que podem ser passados para os construtores e os métodos que você
usa para adicionar conteúdo a elementos e documentos.
Conteúdo válido
As consultas geralmente avaliam para IEnumerable<T> de XElement ou IEnumerable<T> de XAttribute. Você
pode passar coleções de XElement ou de objetos XAttribute para o construtor de XElement. Portanto, é conveniente
passar os resultados de uma consulta como conteúdo em métodos e os construtores que você usa para popular
árvores XML.
Ao adicionar conteúdo simples, vários tipos podem ser passados para esse método. Os tipos válidos incluem:
String
Double
Single
Decimal
Boolean
DateTime
TimeSpan
DateTimeOffset
Qualquer tipo que implemente Object.ToString .
Qualquer tipo que implemente IEnumerable<T>.
Ao adicionar conteúdo complexo, vários tipos podem ser passados para esse método:
XObject
XNode
XAttribute
Qualquer tipo que implemente IEnumerable<T>
Se um objeto implementar IEnumerable<T>, a coleção no objeto será enumerada, e todos os itens da coleção
serão adicionados. Se a coleção contiver objetos XNode ou XAttribute, cada item da coleção será adicionado
separadamente. Se a coleção contiver texto (ou objetos que são convertidos em texto), o texto da coleção será
concatenado e adicionado como um único nó de texto.
Se o conteúdo for null , nada será adicionado. Ao passar itens de um coleção, a coleção pode ser null . Um item
null na coleção não tem nenhum efeito na árvore.
Um atributo adicionado deve ter um nome exclusivo dentro do elemento que o contém.
Ao adicionar objetos XNode ou XAttribute, se o novo conteúdo não tiver um pai, os objetos serão simplesmente
anexados à árvore XML. Se o novo conteúdo já tiver um pai e fizer parte de outra árvore XML, o novo conteúdo
será clonado, e o conteúdo recém-clonado será anexado à árvore XML.
MÉTODO DESCRIÇÃO
Consulte também
Criando árvores XML (C#)
Visão geral sobre namespaces (LINQ to XML)
23/10/2019 • 3 minutes to read • Edit Online
Nomes XML
Os nomes XML são geralmente uma fonte de complexidade na programação XML. Um nome de XML
consiste em um namespace XML (também chamado um URI de um namespace XML ) e um nome local. Um
namespace de XML é semelhante a um namespace em um programa baseadas em .NET Framework. Permite
que você determine exclusivamente os nomes de elementos e atributos. Isso ajuda a evitar conflitos de nome
entre várias partes de um documento XML. Quando você declarar um namespace de XML, poderá selecionar
um nome local que somente deve ser exclusivo dentro desse namespace.
Outro aspecto de nomes XML são os prefixos de namespace de XML. Os prefixos XML causam a maioria da
complexidade de nomes XML. Esses prefixos permite que você crie um atalho para um namespace XML, que
faz o documento XML mais concisas e legível. No entanto, os prefixos XML depende do seu contexto para ter
significado, que adiciona a complexidade. Por exemplo, o prefixo aw XML pode ser associado com a um
namespace XML de uma parte de uma árvore XML, e com um outro namespace XML em uma parte
diferente da árvore XML.
Uma das vantagens de usar LINQ to XML com C# é que você não precisa usar prefixos XML. Quando o
LINQ to XML carrega ou analisa um documento XML, cada prefixo XML é resolvido para o namespace de
XML correspondente. Após isso, quando você trabalha com um documento que usar namespaces, você
acessa quase sempre namespaces com o URI de namespace, e não com o prefixo de namespace. Quando os
desenvolvedores trabalham com nomes XML em LINQ to XML sempre trabalham com um nome XML
totalmente qualificado (isto é, um namespace de XML e um nome local). No entanto, quando necessário,
LINQ to XML permite que você trabalhe com controle e prefixos de namespace.
Em LINQ to XML, a classe que representa nomes XML é XName. Os nomes XML aparecem com frequência
em toda a API LINQ to XML e, em qualquer lugar em que um nome XML for necessário, você encontrará um
parâmetro XName. No entanto, raramente você trabalha diretamente com um XName. XName contém uma
conversão implícita de cadeia de caracteres.
Para obter mais informações, consulte XNamespace e XName.
Como criar um documento com namespaces (C#)
(LINQ to XML)
25/11/2019 • 4 minutes to read • Edit Online
Exemplo
Para criar um elemento ou um atributo que esteja em um namespace, você primeiro declara e inicializa um objeto
XNamespace. Em seguida, você usa a sobrecarga do operador de adição para combinar o namespace com o nome
local, expresso como uma cadeia de caracteres.
O exemplo a seguir cria um documento com um namespace. Por padrão, o LINQ to XML serializa esse documento
com um namespace padrão.
<Root xmlns="http://www.adventure-works.com">
<Child>child content</Child>
</Root>
Exemplo
O exemplo a seguir cria um documento com um namespace. Também cria um atributo que declara o namespace
com um prefixo de namespace. Para criar um atributo que declare um namespace com um prefixo, você cria um
atributo onde o namespace do nome do atributo seja o prefixo do namespace, e esse nome esteja no namespace
Xmlns. O valor desse atributo é o URI do namespace.
<aw:Root xmlns:aw="http://www.adventure-works.com">
<aw:Child>child content</aw:Child>
</aw:Root>
Exemplo
O exemplo a seguir mostra a criação de um documento que contém dois namespaces. Um é o namespace padrão.
O outro é um namespace com um prefixo.
Com a inclusão de atributos de namespace no elemento raiz, os namespaces são serializados, de modo que
http://www.adventure-works.com seja o namespace padrão e www.fourthcoffee.com seja serializado com o prefixo
"fc". Para criar um atributo que declare um namespace padrão, você cria um atributo com o nome "xmlns", sem um
namespace. O valor do atributo é o URI padrão do namespace.
Exemplo
O exemplo a seguir cria um documento que contém dois namespaces, ambos com prefixos de namespace.
XNamespace aw = "http://www.adventure-works.com";
XNamespace fc = "www.fourthcoffee.com";
XElement root = new XElement(aw + "Root",
new XAttribute(XNamespace.Xmlns + "aw", aw.NamespaceName),
new XAttribute(XNamespace.Xmlns + "fc", fc.NamespaceName),
new XElement(fc + "Child",
new XElement(aw + "DifferentChild", "other content")
),
new XElement(aw + "Child2", "c2 content"),
new XElement(fc + "Child3", "c3 content")
);
Console.WriteLine(root);
Exemplo
Outra maneira de obter o mesmo resultado é usar nomes expandidos, em vez de declarar e criar um objeto
XNamespace.
Essa abordagem tem implicações de desempenho. Cada vez que você passa uma cadeia de caracteres que contém
um nome expandido para o LINQ to XML, LINQ to XML deve analisar o nome, localizar o namespace atomizado e
localizar o nome atomizado. Esse processo utiliza tempo de CPU. Se o desempenho for importante, convém
declarar e usar um objeto XNamespace explicitamente.
Se o desempenho for uma questão importante, consulte Pré-atomização de objetos XName (LINQ to XML ) (C#)
para obter mais informações
<aw:Root xmlns:aw="http://www.adventure-works.com">
<aw:Child>child content</aw:Child>
</aw:Root>
Consulte também
Visão geral sobre namespaces (LINQ to XML ) (C#)
Como controlar prefixos de namespaceC#() (LINQ to
XML)
25/11/2019 • 2 minutes to read • Edit Online
Este tópico descreve como você pode controlar prefixos de namespace ao serializar uma árvore XML.
Em muitas situações, não é necessário controlar prefixos de namespace.
No entanto, determinadas ferramentas de programação XML requerem controle específico de prefixos de
namespace. Por exemplo, você pode manipular uma folha de estilos XSLT ou um documento XAML que contenha
as expressões XPath inseridas que se referem a prefixos de namespace específicos. Nesse caso, é importante que o
documento seja serializado com esses prefixos específicos.
Esse é o motivo mais comum para controlar prefixos de namespace.
Outra razão comum para controlar prefixos de namespace é que você deseja que os usuários editem manualmente
o documento XML e deseja criar prefixos de namespace que sejam convenientes para o usuário digitar. Por
exemplo, você pode estar gerando um documento XSD. As convenções de esquemas sugerem que você use xs ou
xsd como o prefixo do namespace do esquema.
Para controlar prefixos de namespace, você insere os atributos que declaram os namespaces. Se você declarar
namespaces com prefixos específicos, o LINQ to XML tentará respeitar os prefixos de namespace ao serializar.
Para criar um atributo que declare um namespace com um prefixo, você cria um atributo onde o namespace do
nome do atributo é Xmlns, e o nome do atributo é o prefixo do namespace. O valor do atributo é o URI do
namespace.
Exemplo
Esse exemplo declara dois namespaces. Ele especifica que o namespace http://www.adventure-works.com tem o
prefixo aw e que o namespace www.fourthcoffee.com tem o prefixo fc .
XNamespace aw = "http://www.adventure-works.com";
XNamespace fc = "www.fourthcoffee.com";
XElement root = new XElement(aw + "Root",
new XAttribute(XNamespace.Xmlns + "aw", "http://www.adventure-works.com"),
new XAttribute(XNamespace.Xmlns + "fc", "www.fourthcoffee.com"),
new XElement(fc + "Child",
new XElement(aw + "DifferentChild", "other content")
),
new XElement(aw + "Child2", "c2 content"),
new XElement(fc + "Child3", "c3 content")
);
Console.WriteLine(root);
Consulte também
Visão geral sobre namespaces (LINQ to XML ) (C#)
Escopo de namespace padrão em C#
23/10/2019 • 2 minutes to read • Edit Online
Namespaces padrões como representadas na árvore XML não estiver no escopo para consultas. Se você tiver XML
que é em um namespace padrão, você ainda deve declarar uma variável de XNamespace , e combina-o com o
nome local para fazer um nome qualificado para ser usado na consulta.
Um dos problemas mais comuns para o consulte árvores XML é que se a árvore tem um namespace XML padrão,
o desenvolvedor escreve às vezes a consulta como se o XML não estar em um namespace.
Definir primeiro exemplos neste tópico mostra uma maneira comum que XML em um namespace padrão é
carregado, mas é visto de modo inadequado.
O segundo conjunto de exemplos a seguir mostra as correções necessárias para que você possa ver XML em um
namespace.
Exemplo
Este exemplo mostra como criar XML em um namespace, e uma consulta que retorna um conjunto de resultados
vazia.
Código
Comentários
Este exemplo gerencia o resultado seguinte:
Exemplo
Este exemplo mostra como criar XML em um namespace, e uma consulta que é codificado corretamente.
Em contraste com incorretamente codificado o exemplo anterior, a abordagem correta para usar C# é declarar e
inicializar um objeto de XNamespace , e usá-lo para especificar XName objetos. Nesse caso, o argumento para o
método de Elements é um objeto de XName .
Código
Comentários
Este exemplo gerencia o resultado seguinte:
Consulte também
Visão geral sobre namespaces (LINQ to XML ) (C#)
Como: Escrever consultas no XML em namespaces
(C#)
23/10/2019 • 2 minutes to read • Edit Online
Para escrever uma consulta em XML que está em um namespace, você deve usar os objetos XName que têm o
namespace correto.
Para C#, a abordagem mais comum é inicializar um XNamespace usando uma cadeia de caracteres que contém o
URI, em seguida, usar a sobrecarga de operador de adição para combinar o namespace com o nome local.
O primeiro conjunto de exemplos neste tópico mostra como criar uma árvore XML em um namespace padrão. O
segundo conjunto mostra como criar uma árvore XML em um namespace com um prefixo.
Exemplo
O exemplo a seguir cria uma árvore XML que está em um namespace padrão. Ele então recupera uma coleção de
elementos.
XNamespace aw = "http://www.adventure-works.com";
XElement root = XElement.Parse(
@"<Root xmlns='http://www.adventure-works.com'>
<Child>1</Child>
<Child>2</Child>
<Child>3</Child>
<AnotherChild>4</AnotherChild>
<AnotherChild>5</AnotherChild>
<AnotherChild>6</AnotherChild>
</Root>");
IEnumerable<XElement> c1 =
from el in root.Elements(aw + "Child")
select el;
foreach (XElement el in c1)
Console.WriteLine((int)el);
1
2
3
Exemplo
No C#, você escreve consultas da mesma forma independentemente se estiver escrevendo consultas em uma
árvore XML que usa um namespace com um prefixo ou em uma árvore XML com um namespace padrão.
O exemplo a seguir cria uma árvore XML que está em um namespace com um prefixo. Ele então recupera uma
coleção de elementos.
XNamespace aw = "http://www.adventure-works.com";
XElement root = XElement.Parse(
@"<aw:Root xmlns:aw='http://www.adventure-works.com'>
<aw:Child>1</aw:Child>
<aw:Child>2</aw:Child>
<aw:Child>3</aw:Child>
<aw:AnotherChild>4</aw:AnotherChild>
<aw:AnotherChild>5</aw:AnotherChild>
<aw:AnotherChild>6</aw:AnotherChild>
</aw:Root>");
IEnumerable<XElement> c1 =
from el in root.Elements(aw + "Child")
select el;
foreach (XElement el in c1)
Console.WriteLine((int)el);
1
2
3
Consulte também
Visão geral sobre namespaces (LINQ to XML ) (C#)
Preservar espaço em branco para serializar
23/10/2019 • 2 minutes to read • Edit Online
Este tópico descreve como controlar o espaço em branco para serializar uma árvore XML.
Um cenário comum é ler o XML recuado, criar uma árvore XML na memória sem nenhum nó de texto de espaço
em branco (isto é, não preservar espaço em branco), executar algumas operações no XML e, em seguida, salvar o
XML com recuo. Quando você serializa o XML com formatação, somente os espaços em branco significativos na
árvore XML são preservados. Este é o comportamento padrão para LINQ to XML.
Outro cenário comum é ler e modificar XML que já foi recuado intencionalmente. Você pode não querer modificar
este recuo de nenhuma forma. Para fazer isso em LINQ to XML, você preserva o espaço em branco quando você
carregar ou analisar XML e desativa o formatação quando você serializa XML.
Este tópico descreve como controlar se a serialização gera uma declaração XML.
<Root><Child>child content</Child></Root>
Consulte também
Serializando árvores XML (C#)
Serializando arquivos, o TextWriters, e o XmlWriters
23/10/2019 • 2 minutes to read • Edit Online
Consulte também
Serializando árvores XML (C#)
Serializando para um XmlReader (invocando XSLT)
(C#)
23/10/2019 • 2 minutes to read • Edit Online
Quando você usa os recursos de interoperabilidade de System.Xml de LINQ to XML, você pode usar CreateReader
para criar um XmlReader. O módulo que lê deste XmlReader lê os nós da árvore XML e processa-os de acordo.
Console.WriteLine(newTree);
<Root>
<C1>Child1 data</C1>
<C2>Child2 data</C2>
</Root>
Consulte também
Serializando árvores XML (C#)
Visão geral dos eixos do LINQ to XML (C#)
04/11/2019 • 6 minutes to read • Edit Online
Após criar uma árvore XML ou carregar um documento XML em uma árvore XML, você poderá consultá-la para
localizar elementos e atributos, e recuperar seus valores. Você recupera coleções por meio dos métodos de eixo,
também denominados eixos. Alguns eixos são métodos nas classes XElement e XDocument que retornam
coleções IEnumerable<T>. Alguns eixos são métodos de extensão na classe Extensions. Os eixos implementados
como métodos de extensão operam em coleções e retornam coleções.
Conforme descrito em Visão geral da classe XElement, um objeto XElement representa um nó de um único
elemento. O conteúdo de um elemento pode ser complexo (às vezes denominado conteúdo estruturado) ou pode
ser um elemento simples. Um elemento simples pode ser vazio ou conter um valor. Se o nó contiver o conteúdo
estruturado, você poderá usar os vários métodos de eixo para recuperar enumerações de elementos
descendentes. Os métodos de eixo mais usados do eixo são Elements e Descendants.
Além dos métodos de eixo, que retornam coleções, há mais dois métodos que você geralmente usa nas
consultas do LINQ to XML. O método Element retorna um XElement único. O método Attribute retorna um
XAttribute único.
Em várias circunstâncias, as consultas do LINQ fornecem a maneira mais eficiente de examinar uma árvore,
extrair dados dela e transformá-la. As consultas do LINQ operam em objetos que implementam o
IEnumerable<T>, e os eixos do LINQ to XML retornam o IEnumerable<T> das coleções XElement e o
IEnumerable<T> das coleções XAttribute. Você precisa dessas coleções para executar suas consultas.
Além dos métodos de eixo que recuperam coleções de elementos e atributos, há métodos de eixo que permitem
a você iterar na árvore detalhadamente. Por exemplo, em vez de tratar elementos e atributos, você pode trabalhar
com os nós da árvore. Os nós são um nível mais refinado de granularidade do que os elementos e os atributos.
Ao trabalhar com os nós, você pode examinar comentários XML, nós de texto, instruções de processamento e
muito mais. Essa funcionalidade é importante, por exemplo, para alguém que estiver escrevendo em um
processador de texto e deseja salvar documentos como XML. No entanto, a maioria dos programadores XML se
preocupam basicamente com os elementos, os atributos e seus valores.
MÉTODO DESCRIÇÃO
MÉTODO DESCRIÇÃO
MÉTODO DESCRIÇÃO
MÉTODO DESCRIÇÃO
Consulte também
Eixos do LINQ to XML (C#)
Como: Recuperar uma coleção de elementos (LINQ
to XML) (C#)
23/10/2019 • 2 minutes to read • Edit Online
Este tópico demonstra o método de Elements . Esse método retorna uma coleção de elementos filho de um
elemento.
Exemplo
Este exemplo efetua iterações através dos elementos filho do elemento de purchaseOrder .
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: Ordem de compra típica (LINQ to XML ).
XElement po = XElement.Load("PurchaseOrder.xml");
IEnumerable<XElement> childElements =
from el in po.Elements()
select el;
foreach (XElement el in childElements)
Console.WriteLine("Name: " + el.Name);
Name: Address
Name: Address
Name: DeliveryNotes
Name: Items
Consulte também
Eixos do LINQ to XML (C#)
Como: Recuperar o valor de um elemento (LINQ to
XML) (C#)
23/10/2019 • 5 minutes to read • Edit Online
Este tópico mostra como obter o valor de elementos. Há duas maneiras principais de fazer isso. Uma maneira é
converter XElement ou XAttribute para o tipo desejado. O operador de conversão explícita converte o conteúdo do
elemento ou do atributo no tipo especificado e o atribui à sua variável. Outra opção é usar a propriedade
XElement.Value ou a propriedade XAttribute.Value.
Entretanto, com C#, a conversão geralmente é a melhor abordagem. Se você converter o elemento ou o atributo
em um tipo anulável, o código será mais simples de criar ao recuperar o valor de um elemento (ou de um atributo)
que pode ou não existir. O último exemplo deste tópico demonstra isso. No entanto, você não pode definir o
conteúdo de um elemento por meio de conversão, como faria usando a propriedade XElement.Value.
Exemplo
Para recuperar o valor de um elemento, basta converter o objeto XElement no tipo desejado. Você sempre pode
converter um elemento em uma cadeia de caracteres, como a seguir:
<StringElement>abcde</StringElement>
Value of e:abcde
Exemplo
Você também pode converter elementos em tipos que não sejam cadeias de caracteres. Por exemplo, se você tiver
um elemento que contenha um número inteiro, poderá convertê-lo em int , como mostrado no código a seguir:
<Age>44</Age>
Value of e:44
LINQ to XML fornece operadores de conversão explícita para os seguintes tipos de dados: string , bool , bool? ,
int , int? , uint , uint? , long , long? , ulong , ulong? , float , float? , double , double? , decimal , decimal? ,
DateTime , DateTime? , TimeSpan , TimeSpan? , GUID e GUID? .
<StringElement>abcde</StringElement>
Value of e:abcde
Exemplo
Às vezes, você tenta recuperar o valor de um elemento mesmo quando não tem certeza de que ele existe. Nesse
caso, quando você atribuir o elemento convertido a um tipo que permite valor nulo ( string ou um elemento dos
tipos que permitem valor nulo no .NET Framework), se o elemento não existir, a variável atribuída será definida
como null . O código a seguir mostra que quando o elemento pode ou não existir, é mais fácil de usar a conversão
do que usar a propriedade Value.
XElement root = new XElement("Root",
new XElement("Child1", "child 1 content"),
new XElement("Child2", "2")
);
string c1 = (string)root.Element("Child1");
Console.WriteLine("c1:{0}", c1 == null ? "element does not exist" : c1);
int? c2 = (int?)root.Element("Child2");
Console.WriteLine("c2:{0}", c2 == null ? "element does not exist" : c2.ToString());
string c3 = (string)root.Element("Child3");
Console.WriteLine("c3:{0}", c3 == null ? "element does not exist" : c3);
int? c4 = (int?)root.Element("Child4");
Console.WriteLine("c4:{0}", c4 == null ? "element does not exist" : c4.ToString());
Console.WriteLine();
XElement e1 = root.Element("Child1");
string v1;
if (e1 == null)
v1 = null;
else
v1 = e1.Value;
Console.WriteLine("v1:{0}", v1 == null ? "element does not exist" : v1);
XElement e2 = root.Element("Child2");
int? v2;
if (e2 == null)
v2 = null;
else
v2 = Int32.Parse(e2.Value);
Console.WriteLine("v2:{0}", v2 == null ? "element does not exist" : v2.ToString());
XElement e3 = root.Element("Child3");
string v3;
if (e3 == null)
v3 = null;
else
v3 = e3.Value;
Console.WriteLine("v3:{0}", v3 == null ? "element does not exist" : v3);
XElement e4 = root.Element("Child4");
int? v4;
if (e4 == null)
v4 = null;
else
v4 = Int32.Parse(e4.Value);
Console.WriteLine("v4:{0}", v4 == null ? "element does not exist" : v4.ToString());
v1:child 1 content
v2:2
v3:element does not exist
v4:element does not exist
Em geral, você pode criar um código mais simples ao usar a conversão para recuperar o conteúdo de elementos e
atributos.
Consulte também
Eixos do LINQ to XML (C#)
Como filtrar em nomes de elementos (LINQ to XML)
(C#)
25/11/2019 • 2 minutes to read • Edit Online
Quando você chamar um dos métodos que IEnumerable<T> de retorno de XElement, você pode filtrar no nome
do elemento.
Exemplo
Este exemplo retorna uma coleção de descendentes que é filtrada para conter somente descendentes com o nome
especificado.
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: ordem de compra típica (LINQ to XML ).
XElement po = XElement.Load("PurchaseOrder.xml");
IEnumerable<XElement> items =
from el in po.Descendants("ProductName")
select el;
foreach(XElement prdName in items)
Console.WriteLine(prdName.Name + ":" + (string) prdName);
ProductName:Lawnmower
ProductName:Baby Monitor
Os outros métodos que IEnumerable<T> de retorno de coleções de XElement segue o mesmo padrão. Suas
assinaturas são semelhantes a Elements e a Descendants. O seguinte é a lista completa dos métodos semelhantes
que tenham assinaturas de método:
Ancestors
Descendants
Elements
ElementsAfterSelf
ElementsBeforeSelf
AncestorsAndSelf
DescendantsAndSelf
Exemplo
O exemplo a seguir mostra a mesma consulta para XML que está em um namespace. Para obter mais informações,
consulte Visão geral de namespaces (LINQ to XML ) (C#).
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: ordem de compra típica em um
namespace.
XNamespace aw = "http://www.adventure-works.com";
XElement po = XElement.Load("PurchaseOrderInNamespace.xml");
IEnumerable<XElement> items =
from el in po.Descendants(aw + "ProductName")
select el;
foreach (XElement prdName in items)
Console.WriteLine(prdName.Name + ":" + (string)prdName);
{http://www.adventure-works.com}ProductName:Lawnmower
{http://www.adventure-works.com}ProductName:Baby Monitor
Consulte também
Eixos do LINQ to XML (C#)
Como encadear chamadas de método de eixo (LINQ
to XMLC#) ()
25/11/2019 • 3 minutes to read • Edit Online
Um padrão comum que você usar em seu código é chamar um método do eixo, então chama um dos eixos do
método de extensão.
Há dois eixos com o nome de Elements que retornam uma coleção de elementos: o método de
XContainer.Elements e o método de Extensions.Elements . Você pode combinar esses dois eixos para localizar
todos os elementos de um nome especificado em uma determinada profundidade na árvore.
Exemplo
Este exemplo usa XContainer.Elements e Extensions.Elements para localizar todos os elementos de Name em todos
os elementos de Address em todos os elementos de PurchaseOrder .
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: vários pedidos de compra (LINQ to
XML ).
<Name>Ellen Adams</Name>
<Name>Tai Yee</Name>
<Name>Cristian Osorio</Name>
<Name>Cristian Osorio</Name>
<Name>Jessica Arnold</Name>
<Name>Jessica Arnold</Name>
Isto funciona porque uma das implementações do eixo de Elements é como um método de extensão em
IEnumerable<T> de XContainer. XElement deriva de XContainer, então você pode chamar o método de
Extensions.Elements nos resultados de uma chamada para o método de XContainer.Elements .
Exemplo
Às vezes você deseja recuperar todos os elementos em uma profundidade específico do elemento quando pode ou
não pode haver ancestrais interveniente. Por exemplo, o seguinte documento, você pode desejar recuperar todos os
elementos de ConfigParameter que são filhos do elemento de Customer , mas não ConfigParameter que é um filho
do elemento de Root .
<Root>
<ConfigParameter>RootConfigParameter</ConfigParameter>
<Customer>
<Name>Frank</Name>
<Config>
<ConfigParameter>FirstConfigParameter</ConfigParameter>
</Config>
</Customer>
<Customer>
<Name>Bob</Name>
<!--This customer doesn't have a Config element-->
</Customer>
<Customer>
<Name>Bill</Name>
<Config>
<ConfigParameter>SecondConfigParameter</ConfigParameter>
</Config>
</Customer>
</Root>
Para fazer isso, você pode usar o eixo de Extensions.Elements , como segue:
<ConfigParameter>FirstConfigParameter</ConfigParameter>
<ConfigParameter>SecondConfigParameter</ConfigParameter>
Exemplo
O exemplo a seguir mostra a mesma técnica para XML que é em um namespace. Para obter mais informações,
consulte Visão geral de namespaces (LINQ to XML ) (C#).
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: vários pedidos de compra em um
namespace.
XNamespace aw = "http://www.adventure-works.com";
XElement purchaseOrders = XElement.Load("PurchaseOrdersInNamespace.xml");
IEnumerable<XElement> names =
from el in purchaseOrders
.Elements(aw + "PurchaseOrder")
.Elements(aw + "Address")
.Elements(aw + "Name")
select el;
foreach (XElement e in names)
Console.WriteLine(e);
Consulte também
Eixos do LINQ to XML (C#)
Como: Recuperar um único elemento filho (LINQ to
XML) (C#)
23/10/2019 • 2 minutes to read • Edit Online
Este tópico explica como recuperar um único elemento filho, considerando o nome do elemento filho. Quando
você souber que o nome do elemento filho e que há apenas um elemento com esse nome, pode ser conveniente
recuperar apenas um elemento, em vez de uma coleção.
O método Element retorna o primeiro filho XElement com o XName especificado.
Se você quiser recuperar um único elemento filho no Visual Basic, uma abordagem comum é usar a propriedade
XML e, em seguida, recuperar o primeiro elemento usando a notação do indexador de matriz.
Exemplo
O exemplo a seguir demonstra o uso do método Element. Este exemplo usa a árvore XML chamada po e localiza
o primeiro elemento chamado Comment .
O exemplo do Visual Basic mostra o uso da notação do indexador de matriz para recuperar um único elemento.
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: Ordem de compra típica (LINQ to XML ).
XElement po = XElement.Load("PurchaseOrder.xml");
XElement e = po.Element("DeliveryNotes");
Console.WriteLine(e);
Exemplo
O exemplo a seguir mostra o mesmo código para XML que está em um namespace. Para obter mais informações,
consulte Visão geral de namespaces (LINQ to XML ) (C#).
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: Ordem de compra típica em um
namespace.
XElement po = XElement.Load("PurchaseOrderInNamespace.xml");
XNamespace aw = "http://www.adventure-works.com";
XElement e = po.Element(aw + "DeliveryNotes");
Console.WriteLine(e);
Este tópico apresenta o método de Attributes . Esse método recupera os atributos de um elemento.
Exemplo
O exemplo a seguir mostra como iterar através da coleção de atributos de um elemento.
ID="1243"
Type="int"
ConvertableTo="double"
Consulte também
Eixos do LINQ to XML (C#)
Como: Recuperar um único atributo (LINQ to XML)
(C#)
23/10/2019 • 2 minutes to read • Edit Online
Este tópico explica como recuperar um único atributo de um elemento, dado o nome do atributo. Isso é útil para
gravar as expressões de consulta onde você deseja localizar um elemento que possui um atributo específico.
O método de Attribute da classe de XElement retorna XAttribute com o nome especificado.
Exemplo
O exemplo a seguir usa o método Attribute.
Este exemplo localiza os descendentes na árvore chamada Phone , e localiza o atributo chamado type .
Esse código gera a seguinte saída:
home
work
Exemplo
Se você deseja recuperar o valor do atributo, você pode convertê-lo, exatamente como você faz para com objetos
de XElement . O exemplo a seguir demonstra isso.
home
work
LINQ to XML fornece operadores cast explícitos para a classe XAttribute para string , bool , bool? , int , int? ,
uint , uint? , long , long? , ulong , ulong? , float , float? , double , double? , decimal , decimal? , DateTime ,
DateTime? , TimeSpan , TimeSpan? , GUID e GUID? .
Exemplo
O exemplo a seguir mostra o mesmo código para um atributo que está em um namespace. Para obter mais
informações, consulte Visão geral de namespaces (LINQ to XML ) (C#).
XNamespace aw = "http://www.adventure-works.com";
XElement cust = new XElement(aw + "PhoneNumbers",
new XElement(aw + "Phone",
new XAttribute(aw + "type", "home"),
"555-555-5555"),
new XElement(aw + "Phone",
new XAttribute(aw + "type", "work"),
"555-555-6666")
);
IEnumerable<XElement> elList =
from el in cust.Descendants(aw + "Phone")
select el;
foreach (XElement el in elList)
Console.WriteLine((string)el.Attribute(aw + "type"));
home
work
Consulte também
Eixos do LINQ to XML (C#)
Como: Recuperar o valor de um atributo (LINQ to
XML) (C#)
23/10/2019 • 2 minutes to read • Edit Online
Este tópico mostra como obter o valor de atributos. Há duas maneiras principais: Você pode converter um
XAttribute no tipo desejado; o operador de conversão explícita converte então o conteúdo do elemento ou do
atributo no tipo especificado. Outra opção é usar a propriedade Value. No entanto, a conversão geralmente é a
abordagem recomendada. Se você converter o atributo em um tipo anulável, o código será mais simples de criar
ao recuperar o valor de um atributo que pode ou não existir. Para obter exemplos dessa técnica, confira Como:
Recuperar o valor de um elemento (LINQ to XML ) (C#).
Exemplo
Para recuperar o valor de um atributo, basta converter o objeto XAttribute no tipo desejado.
Exemplo
O exemplo a seguir mostra como recuperar o valor de um atributo em que o atributo está em um namespace. Para
obter mais informações, consulte Visão geral de namespaces (LINQ to XML ) (C#).
XNamespace aw = "http://www.adventure-works.com";
XElement root = new XElement(aw + "Root",
new XAttribute(aw + "Attr", "abcde")
);
string str = (string)root.Attribute(aw + "Attr");
Console.WriteLine(str);
abcde
Consulte também
Eixos do LINQ to XML (C#)
Como: Recuperar o valor superficial de um elemento
(C#)
23/10/2019 • 2 minutes to read • Edit Online
Este tópico mostra como obter o valor raso de um elemento. O valor raso é o valor do elemento específico
somente, diferentemente de valor maior, que inclui os valores de todos os elementos descendentes concatenados
em uma única cadeia de caracteres.
Quando você recupera um valor de elemento usando conversão ou propriedade de XElement.Value , você recupera
o valor maior. Para recuperar o valor raso, você pode usar o método de extensão de ShallowValue , conforme
mostrado no exemplo a seguir. Recuperar o valor raso é útil quando você deseja selecionar elementos com base no
conteúdo.
O exemplo a seguir declara um método de extensão que recupera o valor raso de um elemento. Use o método de
extensão em uma consulta para listar todos os elementos que contém um valor calculado.
Exemplo
O seguinte arquivo de texto, Report.xml, é a fonte para esse exemplo.
class Program
{
static void Main(string[] args)
{
XElement root = XElement.Load("Report.xml");
Consulte também
Eixos do LINQ to XML (C#)
Como localizar um elemento com um atributo
específico (C#)
25/11/2019 • 2 minutes to read • Edit Online
Este tópico mostra como localizar um elemento que tem um atributo que tem um valor específico.
Exemplo
O exemplo mostra como localizar o elemento Address que tem um atributo Type com um valor de
“faturamento”.
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: ordem de compra típica (LINQ to XML ).
<Address Type="Billing">
<Name>Tai Yee</Name>
<Street>8 Oak Avenue</Street>
<City>Old Town</City>
<State>PA</State>
<Zip>95819</Zip>
<Country>USA</Country>
</Address>
Exemplo
O exemplo a seguir mostra a mesma consulta para XML que está em um namespace. Para obter mais
informações, consulte Visão geral de namespaces (LINQ to XML ) (C#).
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: ordem de compra típica em um
namespace.
Consulte também
Attribute
Elements
Visão geral de operadores de consulta padrão (C#)
Operações de projeção (C#)
Como localizar um elemento com um elemento filho
específico (C#)
25/11/2019 • 2 minutes to read • Edit Online
Este tópico mostra como localizar determinado elemento que tem um elemento filho com um valor específico.
Exemplo
O exemplo localiza o elemento Test que possui um elemento filho CommandLine com o valor "Examp2.EXE".
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: configuração de teste (LINQ to XML ).
0002
0006
Exemplo
O exemplo a seguir mostra a mesma consulta para XML que está em um namespace. Para obter mais informações,
consulte Visão geral de namespaces (LINQ to XML ) (C#).
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: configuração de teste em um namespace.
0002
0006
Consulte também
Attribute
Elements
Visão geral de operadores de consulta padrão (C#)
Operações de projeção (C#)
Consultar um XDocument versus Consultar um
XElement (C#)
23/10/2019 • 2 minutes to read • Edit Online
Ao carregar um documento por meio do XDocument.Load, você observará que precisa escrever consultas um
pouco diferentes do que ao carregar por meio do XElement.Load.
O exemplo a seguir é o mesmo que o anterior, com a exceção de que a árvore XML é carregada em um XDocument
em vez de em um XElement.
// Create a simple document and write it to a file
File.WriteAllText("Test.xml", @"<Root>
<Child1>1</Child1>
<Child2>2</Child2>
<Child3>3</Child3>
</Root>");
Observe que a mesma consulta retornou o nó Root em vez dos três nós filho.
Uma abordagem para lidar com isso é usar a propriedade Root antes de acessar os métodos de eixos, da seguinte
maneira:
Essa consulta agora executa da mesma maneira que a consulta na árvore enraizada em XElement. O exemplo
produz a seguinte saída:
Às vezes, você deseja localizar todos os descendentes com um nome específico. Você poderia escrever um código
para iterar por todos os descendentes, mas é mais fácil usar o eixo Descendants.
Exemplo
O exemplo a seguir mostra como localizar os descendentes com base no nome do elemento.
Console.WriteLine(str);
Exemplo
O exemplo a seguir mostra a mesma consulta para XML que está em um namespace. Para obter mais informações,
consulte Visão geral de namespaces (LINQ to XML ) (C#).
XElement root = XElement.Parse(@"<root xmlns='http://www.adatum.com'>
<para>
<r>
<t>Some text </t>
</r>
<n>
<r>
<t>that is broken up into </t>
</r>
</n>
<n>
<r>
<t>multiple segments.</t>
</r>
</n>
</para>
</root>");
XNamespace ad = "http://www.adatum.com";
IEnumerable<string> textSegs =
from seg in root.Descendants(ad + "t")
select (string)seg;
Console.WriteLine(str);
Consulte também
Descendants
Como localizar um único descendente usando o
método descendentes (C#)
25/11/2019 • 2 minutes to read • Edit Online
Você pode usar o método de eixo Descendants para rapidamente escrever código para localizar um único elemento
nomeado exclusivamente. Essa técnica é especialmente útil quando você quer localizar um descendente específico
com um nome específico. Você pode escrever o código para navegar para o elemento desejado, mas geralmente é
mais rápido e fácil escrever código usando o eixo Descendants.
Exemplo
Este exemplo usa o operador padrão de consulta First.
GC3 Value
Exemplo
O exemplo a seguir mostra a mesma consulta para XML que está em um namespace. Para obter mais informações,
consulte Visão geral de namespaces (LINQ to XML ) (C#).
XElement root = XElement.Parse(@"<aw:Root xmlns:aw='http://www.adventure-works.com'>
<aw:Child1>
<aw:GrandChild1>GC1 Value</aw:GrandChild1>
</aw:Child1>
<aw:Child2>
<aw:GrandChild2>GC2 Value</aw:GrandChild2>
</aw:Child2>
<aw:Child3>
<aw:GrandChild3>GC3 Value</aw:GrandChild3>
</aw:Child3>
<aw:Child4>
<aw:GrandChild4>GC4 Value</aw:GrandChild4>
</aw:Child4>
</aw:Root>");
XNamespace aw = "http://www.adventure-works.com";
string grandChild3 = (string)
(from el in root.Descendants(aw + "GrandChild3")
select el).First();
Console.WriteLine(grandChild3);
GC3 Value
Como: Escrever consultas com filtragem complexa
(C#)
23/10/2019 • 2 minutes to read • Edit Online
Muitas vezes, você deseja escrever consultas LINQ to XML com filtros complexos. Por exemplo, você pode ter que
localizar todos os elementos que têm um elemento filho com um nome e um valor específicos. Este tópico dá um
exemplo de como escrever uma consulta com filtragem complexa.
Exemplo
Este exemplo mostra como localizar todos os elementos PurchaseOrder que têm um elemento filho Address que
têm um atributo Type igual a “Shipping” e um elemento filho State igual a “NY”. Ele usa uma consulta aninhada
na cláusula Where e o operador Any retornará true se a coleção tiver elementos nele. Para obter informações
sobre como usar a sintaxe de consulta baseada em método, consulte Sintaxe de consulta e sintaxe de método em
LINQ.
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: Várias ordens de compra (LINQ to XML ).
Para obter mais informações sobre o operador Any , consulte Operações de quantificador (C#).
99505
Exemplo
O exemplo a seguir mostra a mesma consulta para XML que está em um namespace. Para obter mais
informações, consulte Visão geral de namespaces (LINQ to XML ) (C#).
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: Várias ordens de compra em um
namespace.
XElement root = XElement.Load("PurchaseOrdersInNamespace.xml");
XNamespace aw = "http://www.adventure-works.com";
IEnumerable<XElement> purchaseOrders =
from el in root.Elements(aw + "PurchaseOrder")
where
(from add in el.Elements(aw + "Address")
where
(string)add.Attribute(aw + "Type") == "Shipping" &&
(string)add.Element(aw + "State") == "NY"
select add)
.Any()
select el;
foreach (XElement el in purchaseOrders)
Console.WriteLine((string)el.Attribute(aw + "PurchaseOrderNumber"));
99505
Consulte também
Attribute
Elements
Operações de projeção (C#)
Operações de quantificador (C#)
Como filtrar em um elemento opcional (C#)
25/11/2019 • 2 minutes to read • Edit Online
Às vezes você deseja filtrar um elemento mesmo que você não tenha certeza ele existe em seu documento XML. A
pesquisa deve ser executada de modo que se o elemento específico não tem o elemento filho, você não dispare
uma exceção de referência nula filtragem para ele. No exemplo a seguir, o elemento de Child5 não tiver um
elemento filho de Type , mas a consulta ainda executa corretamente.
Exemplo
Este exemplo usa o método de extensão de Elements .
Exemplo
O exemplo a seguir mostra a mesma consulta para XML que está em um namespace. Para obter mais informações,
consulte Visão geral de namespaces (LINQ to XML ) (C#).
XElement root = XElement.Parse(@"<Root xmlns='http://www.adatum.com'>
<Child1>
<Text>Child One Text</Text>
<Type Value=""Yes""/>
</Child1>
<Child2>
<Text>Child Two Text</Text>
<Type Value=""Yes""/>
</Child2>
<Child3>
<Text>Child Three Text</Text>
<Type Value=""No""/>
</Child3>
<Child4>
<Text>Child Four Text</Text>
<Type Value=""Yes""/>
</Child4>
<Child5>
<Text>Child Five Text</Text>
</Child5>
</Root>");
XNamespace ad = "http://www.adatum.com";
var cList =
from typeElement in root.Elements().Elements(ad + "Type")
where (string)typeElement.Attribute("Value") == "Yes"
select (string)typeElement.Parent.Element(ad + "Text");
foreach (string str in cList)
Console.WriteLine(str);
Consulte também
XElement.Attribute
XContainer.Elements
Extensions.Elements
Visão geral de operadores de consulta padrão (C#)
Operações de projeção (C#)
Como localizar todos os nós em um namespace (C#)
25/11/2019 • 2 minutes to read • Edit Online
Você pode filtrar no namespace de cada elemento ou atributo para localizar todos os nós nesse namespace
específico.
Exemplo
O exemplo a seguir cria uma árvore XML com as duas namespaces. Em itera através da árvore e imprime os
nomes de todos os elementos e atributos em uma desses namespaces.
Exemplo
O arquivo XML acessado pela consulta contém pedidos de compra em dois namespaces diferentes. A consulta cria
uma nova árvore com apenas os elementos em uma namespaces.
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: pedidos de compra consolidados.
Este exemplo mostra como escrever uma consulta que classifica seus resultados.
Exemplo
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: Dados numéricos (LINQ to XML ).
0.99
4.95
6.99
24.50
29.00
66.00
89.99
Exemplo
O exemplo a seguir mostra a mesma consulta para XML que está em um namespace. Para obter mais
informações, consulte Visão geral de namespaces (LINQ to XML ) (C#).
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: Dados numéricos em um namespace.
Consulte também
Classificando dados (C#)
Como: Classificar elementos em várias chaves (C#)
23/10/2019 • 2 minutes to read • Edit Online
Exemplo
Nesse exemplo, os resultados são classificados primeiro pelo código postal de enviar, então a data do pedido.
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: Clientes e ordens (LINQ to XML ).
XElement co = XElement.Load("CustomersOrders.xml");
var sortedElements =
from c in co.Element("Orders").Elements("Order")
orderby (string)c.Element("ShipInfo").Element("ShipPostalCode"),
(DateTime)c.Element("OrderDate")
select new {
CustomerID = (string)c.Element("CustomerID"),
EmployeeID = (string)c.Element("EmployeeID"),
ShipPostalCode = (string)c.Element("ShipInfo").Element("ShipPostalCode"),
OrderDate = (DateTime)c.Element("OrderDate")
};
foreach (var r in sortedElements)
Console.WriteLine("CustomerID:{0} EmployeeID:{1} ShipPostalCode:{2} OrderDate:{3:d}",
r.CustomerID, r.EmployeeID, r.ShipPostalCode, r.OrderDate);
Exemplo
O exemplo a seguir mostra a mesma consulta para XML que está em um namespace. Para obter mais
informações, consulte Visão geral de namespaces (LINQ to XML ) (C#).
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: Clientes e ordens em um namespace.
XElement co = XElement.Load("CustomersOrdersInNamespace.xml");
XNamespace aw = "http://www.adventure-works.com";
var sortedElements =
from c in co.Element(aw + "Orders").Elements(aw + "Order")
orderby (string)c.Element(aw + "ShipInfo").Element(aw + "ShipPostalCode"),
(DateTime)c.Element(aw + "OrderDate")
select new
{
CustomerID = (string)c.Element(aw + "CustomerID"),
EmployeeID = (string)c.Element(aw + "EmployeeID"),
ShipPostalCode = (string)c.Element(aw + "ShipInfo").Element(aw + "ShipPostalCode"),
OrderDate = (DateTime)c.Element(aw + "OrderDate")
};
foreach (var r in sortedElements)
Console.WriteLine("CustomerID:{0} EmployeeID:{1} ShipPostalCode:{2} OrderDate:{3:d}",
r.CustomerID, r.EmployeeID, r.ShipPostalCode, r.OrderDate);
Este exemplo mostra como calcular valores intermediários que podem ser usados na classificação, filtragem, e em
selecionar.
Exemplo
O exemplo a seguir utiliza a cláusula de Let .
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: dados numéricos (LINQ to XML ).
55.92
73.50
89.99
198.00
435.00
Exemplo
O exemplo a seguir mostra a mesma consulta para XML que está em um namespace. Para obter mais
informações, consulte Visão geral de namespaces (LINQ to XML ) (C#).
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: dados numéricos em um namespace.
Muitas vezes você pode ter que escrever uma consulta que seleciona elementos com base no contexto. Você pode
querer filtrar com base nos elementos irmãos precedentes ou seguintes. Você pode querer filtrar com base nos
elementos filhos ou ancestrais.
Você pode fazer isso escrevendo uma consulta e usando os resultados da consulta na cláusula where . Se você
primeiro tiver que testar com zero e, em seguida, testar o valor, é mais conveniente fazer a consulta em uma
cláusula let e usar os resultados na cláusula where .
Exemplo
O exemplo a seguir seleciona todos os elementos p que são imediatamente seguidos por um elemento ul .
IEnumerable<XElement> items =
from e in doc.Descendants("p")
let z = e.ElementsAfterSelf().FirstOrDefault()
where z != null && z.Name.LocalName == "ul"
select e;
id = 1
id = 3
id = 6
Exemplo
O exemplo a seguir mostra a mesma consulta para XML que está em um namespace. Para obter mais informações,
consulte Visão geral de namespaces (LINQ to XML ) (C#).
XElement doc = XElement.Parse(@"<Root xmlns='http://www.adatum.com'>
<p id=""1""/>
<ul>abc</ul>
<Child>
<p id=""2""/>
<notul/>
<p id=""3""/>
<ul>def</ul>
<p id=""4""/>
</Child>
<Child>
<p id=""5""/>
<notul/>
<p id=""6""/>
<ul>abc</ul>
<p id=""7""/>
</Child>
</Root>");
XNamespace ad = "http://www.adatum.com";
IEnumerable<XElement> items =
from e in doc.Descendants(ad + "p")
let z = e.ElementsAfterSelf().FirstOrDefault()
where z != null && z.Name == ad.GetName("ul")
select e;
id = 1
id = 3
id = 6
Consulte também
Parse
Descendants
ElementsAfterSelf
FirstOrDefault
Como depurar conjuntos de resultados de consulta
vaziosC#()
25/11/2019 • 2 minutes to read • Edit Online
Um dos problemas mais comuns para o consulte árvores XML é que se a árvore tem um namespace XML padrão,
o desenvolvedor escreve às vezes a consulta como se o XML não estar em um namespace.
Definir primeiro exemplos neste tópico mostra uma maneira comum que XML em um namespace padrão é
carregado, e deduzido de modo inadequado.
O segundo conjunto de exemplos a seguir mostra as correções necessárias para que você possa ver XML em um
namespace.
Para obter mais informações, consulte Visão geral de namespaces (LINQ to XML ) (C#).
Exemplo
Este exemplo mostra como criar XML em um namespace, e uma consulta que retorna um conjunto de resultados
vazia.
Exemplo
Este exemplo mostra como criar XML em um namespace, e uma consulta que é codificado corretamente.
A solução é declarar e inicializar um objeto de XNamespace e usá-lo para especificar XName objetos. Nesse caso,
o argumento para o método de Elements é um objeto de XName .
XElement root = XElement.Parse(
@"<Root xmlns='http://www.adventure-works.com'>
<Child>1</Child>
<Child>2</Child>
<Child>3</Child>
<AnotherChild>4</AnotherChild>
<AnotherChild>5</AnotherChild>
<AnotherChild>6</AnotherChild>
</Root>");
XNamespace aw = "http://www.adventure-works.com";
IEnumerable<XElement> c1 =
from el in root.Elements(aw + "Child")
select el;
Console.WriteLine("Result set follows:");
foreach (XElement el in c1)
Console.WriteLine((int)el);
Console.WriteLine("End of result set");
É conveniente converter variedades de estruturas de dados para XML, e XML de volta para outras estruturas de
dados. Este tópico mostra uma implementação específica dessa abordagem geral convertendo
Dictionary<TKey,TValue> a XML e verso.
Exemplo
Este exemplo usa um formulário de compilação funcional em que uma consulta projetos novos objetos de
XElement e a coleção resultante é passada como um argumento para o construtor do objeto de XElement raiz.
<Root>
<Child1>Value1</Child1>
<Child2>Value2</Child2>
<Child3>Value3</Child3>
<Child4>Value4</Child4>
</Root>
Exemplo
O código a seguir cria um dicionário XML.
A forma de um documento XML refere-se aos seus nomes de elemento, nomes de atributo e às características de
sua hierarquia.
Às vezes você precisará alterar a forma de um documento XML. Por exemplo, você pode ter que enviar um
documento XML existente para outro sistema que requer nomes diferentes de elementos e atributos. Você pode
examinar o documento, excluir e renomeando elementos conforme necessário, mas usando funcionais resultados
de compilação em um código mais legível e mais sustentável. Para obter mais informações sobre a construção
funcional, consulte Construção funcional (LINQ to XML ) (C#).
O primeiro exemplo altera a organização de documento XML. Mover elementos complexos de um local na árvore
para outro.
O segundo exemplo neste tópico cria um documento XML com uma forma diferente do documento de origem.
Altera a caixa de nomes de elemento, renomear alguns elementos, e permitir que alguns elementos da árvore de
origem fora da árvore transformada.
Exemplo
As seguintes alterações de código a forma de um arquivo XML usando expressões inseridas de consulta.
O documento XML de origem neste exemplo contém um elemento de Customers no elemento de Root que
contém todos os clientes. Também contém um elemento de Orders no elemento de Root que contém todos os
pedidos. Este exemplo cria uma nova árvore XML em que os pedidos para cada cliente estão contidos em um
elemento de Orders dentro do elemento de Customer . O documento original também contém um elemento de
CustomerID no elemento de Order ; este elemento será removido do documento reformulado.
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: Clientes e ordens (LINQ to XML ).
XElement co = XElement.Load("CustomersOrders.xml");
XElement newCustOrd =
new XElement("Root",
from cust in co.Element("Customers").Elements("Customer")
select new XElement("Customer",
cust.Attributes(),
cust.Elements(),
new XElement("Orders",
from ord in co.Element("Orders").Elements("Order")
where (string)ord.Element("CustomerID") == (string)cust.Attribute("CustomerID")
select new XElement("Order",
ord.Attributes(),
ord.Element("EmployeeID"),
ord.Element("OrderDate"),
ord.Element("RequiredDate"),
ord.Element("ShipInfo")
)
)
)
);
Console.WriteLine(newCustOrd);
Exemplo
Este exemplo renomeia alguns elementos e converte alguns atributos para elementos.
O código chama ConvertAddress , que retorna uma lista de objetos XElement . O argumento para o método é uma
consulta que determina o elemento complexo de Address onde o atributo de Type tem um valor de "Shipping" .
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: Ordem de compra típica (LINQ to XML ).
static IEnumerable<XElement> ConvertAddress(XElement add)
{
List<XElement> fragment = new List<XElement>() {
new XElement("NAME", (string)add.Element("Name")),
new XElement("STREET", (string)add.Element("Street")),
new XElement("CITY", (string)add.Element("City")),
new XElement("ST", (string)add.Element("State")),
new XElement("POSTALCODE", (string)add.Element("Zip")),
new XElement("COUNTRY", (string)add.Element("Country"))
};
return fragment;
}
<PO>
<ID>99503</ID>
<DATE>1999-10-20T00:00:00</DATE>
<NAME>Ellen Adams</NAME>
<STREET>123 Maple Street</STREET>
<CITY>Mill Valley</CITY>
<ST>CA</ST>
<POSTALCODE>10999</POSTALCODE>
<COUNTRY>USA</COUNTRY>
</PO>
Como controlar o tipo de uma projeção (C#)
25/11/2019 • 2 minutes to read • Edit Online
A projeção é o processo de receber um dataset, de filtre-o, para alterar sua forma, e mesmo de alterar seu tipo. A
maioria das expressões de consulta executam projeções. A maioria das expressões de consulta mostradas nesta
seção valor para IEnumerable<T> de XElement, mas você pode controlar o tipo de projeção para criar coleções de
outros tipos. Este tópico mostra como fazer isso.
Exemplo
O exemplo a seguir define um novo tipo, Customer . A expressão de consulta cria uma instância em novos objetos
de Customer na cláusula Select . Isso faz com que o tipo da expressão de consulta para ser IEnumerable<T> de
Customer .
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: clientes e pedidos (LINQ to XML ).
class Program
{
static void Main(string[] args)
{
XElement custOrd = XElement.Load("CustomersOrders.xml");
IEnumerable<Customer> custList =
from el in custOrd.Element("Customers").Elements("Customer")
select new Customer(
(string)el.Attribute("CustomerID"),
(string)el.Element("CompanyName"),
(string)el.Element("ContactName")
);
foreach (Customer cust in custList)
Console.WriteLine(cust);
}
}
Consulte também
Select
Como: Projetar um novo tipo (LINQ to XML) (C#)
23/10/2019 • 2 minutes to read • Edit Online
Outros exemplos nesta seção mostraram consultas que os resultados de retorno como IEnumerable<T> de
XElement, IEnumerable<T> de string , e IEnumerable<T> de int . Esses são tipos comuns de resultado, mas não
são adequados para cada cenário. Em muitos casos você desejará suas consultas para retornar IEnumerable<T> de
qualquer outro tipo.
Exemplo
Este exemplo mostra como instanciar objetos na cláusula select . O código define primeiro uma nova classe com
um construtor, e altera a declaração de select de modo que a expressão é uma nova instância da nova classe.
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: Ordem de compra típica (LINQ to XML ).
class NameQty
{
public string name;
public int qty;
public NameQty(string n, int q)
{
name = n;
qty = q;
}
};
class Program {
public static void Main()
{
XElement po = XElement.Load("PurchaseOrder.xml");
IEnumerable<NameQty> nqList =
from n in po.Descendants("Item")
select new NameQty(
(string)n.Element("ProductName"),
(int)n.Element("Quantity")
);
Este exemplo usa o método de Element que foi introduzido no tópico Como: Recuperar um único elemento filho
(LINQ to XML ) (C#). Também usa conversões para recuperar os valores dos elementos que são retornados pelo
método de Element .
Este exemplo gera a seguinte saída:
Lawnmower:1
Baby Monitor:2
Como: Projetar um grafo de objeto (C#)
23/10/2019 • 4 minutes to read • Edit Online
Exemplo
O seguinte código popula um grafo de objeto com as classes Address , PurchaseOrder e PurchaseOrderItem do
documento XML Arquivo XML de exemplo: Ordem de compra típica (LINQ to XML ).
class Address
{
public enum AddressUse
{
Shipping,
Billing,
}
class PurchaseOrderItem
{
private string partNumber;
private string productName;
private int quantity;
private Decimal usPrice;
private string comment;
private DateTime shipDate;
class PurchaseOrder
{
private string purchaseOrderNumber;
private DateTime orderDate;
private string comment;
private List<Address> addresses;
private List<PurchaseOrderItem> items;
class Program {
public static void Main()
{
XElement po = XElement.Load("PurchaseOrder.xml");
PurchaseOrder purchaseOrder = new PurchaseOrder {
PurchaseOrderNumber = (string)po.Attribute("PurchaseOrderNumber"),
OrderDate = (DateTime)po.Attribute("OrderDate"),
Addresses = (
from a in po.Elements("Address")
select new Address {
AddressType = ((string)a.Attribute("Type") == "Shipping") ?
Address.AddressUse.Shipping :
Address.AddressUse.Billing,
Name = (string)a.Element("Name"),
Street = (string)a.Element("Street"),
City = (string)a.Element("City"),
State = (string)a.Element("State"),
Zip = (string)a.Element("Zip"),
Country = (string)a.Element("Country")
}
).ToList(),
Items = (
from i in po.Element("Items").Elements("Item")
select new PurchaseOrderItem {
PartNumber = (string)i.Attribute("PartNumber"),
ProductName = (string)i.Element("ProductName"),
Quantity = (int)i.Element("Quantity"),
USPrice = (Decimal)i.Element("USPrice"),
Comment = (string)i.Element("Comment"),
ShipDate = (i.Element("ShipDate") != null) ?
(DateTime)i.Element("ShipDate") :
(DateTime)i.Element("ShipDate") :
DateTime.MinValue
}
).ToList()
};
Console.WriteLine(purchaseOrder);
}
}
PurchaseOrderNumber: 99503
OrderDate: 10/20/1999
Addresses
=====
Type: Shipping
Name: Ellen Adams
Street: 123 Maple Street
City: Mill Valley
State: CA
Zip: 10999
Country: USA
Type: Billing
Name: Tai Yee
Street: 8 Oak Avenue
City: Old Town
State: PA
Zip: 95819
Country: USA
Items
=====
PartNumber: 872-AA
ProductName: Lawnmower
Quantity: 1
USPrice: 148.95
Comment: Confirm this is electric
PartNumber: 926-AA
ProductName: Baby Monitor
Quantity: 2
USPrice: 39.98
ShipDate: 5/21/1999
Consulte também
Select
ToList
Como: Projetar um tipo anônimo (C#)
23/10/2019 • 2 minutes to read • Edit Online
Em alguns casos você pode querer projetar uma consulta a um novo tipo, mesmo que você soubesse que você
usará apenas este tipo para um curto quando. É muito trabalho adicional para criar apenas um novo tipo para usar
na projeção. Uma abordagem mais eficiente nesse caso é projeto para um tipo anônimo. Tipos anônimos permitem
que você defina uma classe, então declare e inicialize um objeto de aquela classe, sem dar um nome para a classe.
Os tipos anônimos são a implementação de C# do conceito matemático de uma tupla. O tuple o termo matemático
proveniente da sequência única, double, triplo, quádruplo, quintuple, n- tuple. Refere-se a uma sequência finito
rotuladas de objetos, cada um de um tipo específico. Isso é às vezes chamado uma lista de pares nome/valor. Por
exemplo, o conteúdo de um endereço no documento XML Arquivo XML de exemplo: Ordem de compra típica
(LINQ to XML ) pode ser expresso da seguinte maneira:
Quando você cria uma instância de um tipo anônimo, é conveniente pensar nele como criar um tuple de Em ordem.
Se você escrever uma consulta que cria um tuple na cláusula select , a consulta retorna IEnumerable de tuple.
Exemplo
Nesse exemplo, a cláusula de select projetos de um tipo anônimo. O exemplo usa em var para criar o objeto de
IEnumerable . Dentro do loop de foreach , a variável de iteração torna-se em uma instância do tipo anônimo
criado na expressão de consulta.
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: Clientes e ordens (LINQ to XML ).
Este exemplo mostra como gerar um arquivo separado por vírgulas de (CSV ) dos valores de um arquivo XML.
Exemplo
A versão de C# este exemplo usa a sintaxe método e o operador de Aggregate para gerar um arquivo CSV de um
documento XML em uma única expressão. Para obter mais informações, consulte Sintaxe de consulta e sintaxe de
método em LINQ.
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: clientes e pedidos (LINQ to XML ).
Consulte também
Projeções e transformações (LINQ to XML ) (C#)
Como: Gerar um XML com base em arquivos CSV
(C#)
23/10/2019 • 2 minutes to read • Edit Online
Este exemplo mostra como usar LINQ (Consulta Integrada à Linguagem) e LINQ to XML para gerar um arquivo
XML de um arquivo CSV (valores separados por vírgulas).
Exemplo
O código a seguir executa uma consulta LINQ em uma matriz de cadeias de caracteres.
A consulta usa a cláusula let para dividir cada cadeia de caracteres em uma matriz de campos.
Um elemento ou atributo em um documento XML, algumas vezes, pode fazer referência a outro elemento ou
atributo. Por exemplo, o documento XML Arquivo XML de exemplo: Clientes e ordens (LINQ to XML ) contém
uma lista de clientes e uma lista de ordens. Cada elemento Customer contém um atributo CustomerID . Cada
elemento Order contém um atributo CustomerID . O elemento CustomerID em cada pedido faz referência ao
atributo CustomerID em um cliente.
O tópico Arquivo XSD de exemplo: Clientes e ordens contém um XSD que pode ser usado para validar esse
documento. Usa os recursos de XSD xs:key e xs:keyref para estabelecer que o atributo CustomerID do
elemento Customer é uma chave, e para estabelecer uma relação entre o elemento CustomerID em cada elemento
Order , e o atributo CustomerID em cada elemento Customer .
Com o LINQ to XML, você pode tirar proveito dessa relação usando a cláusula join .
Observe que como não existe nenhum índice disponível, essa junção terá um desempenho inadequado em tempo
de execução.
Para obter informações mais detalhadas sobre join , consulte Operações de junção (C#).
Exemplo
O exemplo a seguir une os elementos Customer aos elementos Order e gera um novo documento XML que inclui
o elemento CompanyName nos pedidos.
Antes de executar a consulta, o exemplo valida que o documento está de acordo com o esquema de Arquivo XSD
de exemplo: Clientes e ordens. Isso garante que a cláusula join sempre funcionará.
Esta consulta retorna primeiro todos os elementos Customer e, em seguida, une-os aos elementos Order . A
consulta seleciona apenas os pedidos de clientes com um CustomerID maior que "K". Em seguida, projeta um novo
elemento Order que contém as informações do cliente dentro de cada pedido.
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: Clientes e ordens (LINQ to XML ).
Este exemplo usa o seguinte esquema XSD: Arquivo XSD de exemplo: Clientes e ordens.
Observe que a junção dessa forma não será muito bem-executada. As junções são executadas por meio de uma
pesquisa linear. Não há nenhuma tabela de hash ou índice para ajudar no desempenho.
XmlSchemaSet schemas = new XmlSchemaSet();
schemas.Add("", "CustomersOrders.xsd");
if (!errors)
{
// Join customers and orders, and create a new XML document with
// a different shape.
Este exemplo mostra como agrupar dados, e gerencia em XML baseado em agrupamento.
Exemplo
Este exemplo primeiro agrupa dados por uma categoria, então gerencia um novo arquivo XML na hierarquia XML
reflete o agrupamento.
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: dados numéricos (LINQ to XML ).
Este tópico apresenta os métodos de extensão que permitem ver uma árvore XML usando o XPath. Para obter
informações detalhadas sobre como usar esses métodos de extensão, consulte System.Xml.XPath.Extensions.
A menos que você tenha um motivo específico para muito consulte o XPath em uso, como o uso extensivo de
código herdado, usando o XPath com LINQ to XML não é recomendada. Consultas de XPath não são executadas
tão bem quanto consultas de LINQ to XML.
Exemplo
O exemplo a seguir cria uma árvore XML pequena e usa XPathSelectElements para selecionar um conjunto de
elementos.
<Child2>4</Child2>
<Child2>5</Child2>
<Child2>6</Child2>
Como: Escrever um método de eixo LINQ to XML
(C#)
23/10/2019 • 4 minutes to read • Edit Online
Você pode escrever seus próprios métodos do eixo para recuperar coleções de uma árvore XML. Uma das
melhores maneiras de fazer isso é gravar um método de extensão que retorna uma coleção de elementos ou
atributos. Você pode escrever seu método de extensão para subconjuntos específicos de retorno de elementos ou
atributos, com base nos requisitos do seu aplicativo.
Exemplo
O exemplo a seguir usa dois métodos de extensão. O primeiro método de extensão, GetXPath , opera em XObject, e
retorna uma expressão XPath que quando avaliada retorna o nó ou do atributo. O segundo método de extensão,
Find , opera em XElement. Retorna uma coleção de objetos XAttribute e objetos de XElement que contêm texto
especificado.
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: Várias ordens de compra (LINQ to XML ).
class Program
{
static void Main(string[] args)
{
XElement purchaseOrders = XElement.Load("PurchaseOrders.xml");
IEnumerable<XObject> subset =
from xobj in purchaseOrders.Find("1999")
select xobj;
select xobj;
/PurchaseOrders/PurchaseOrder[1]/@OrderDate
1999-10-20
/PurchaseOrders/PurchaseOrder[1]/Items/Item[2]/ShipDate
1999-05-21
/PurchaseOrders/PurchaseOrder[2]/@OrderDate
1999-10-22
/PurchaseOrders/PurchaseOrder[3]/@OrderDate
1999-10-22
Como: Executar transformações de streaming de
texto para XML (C#)
23/10/2019 • 2 minutes to read • Edit Online
Uma abordagem para processar um arquivo de texto é escrever um método de extensão que passa o arquivo de
texto uma linha em uma hora usando a compilação de yield return . Você então pode escrever uma consulta
LINQ que processa o arquivo de texto em uma forma adiada lazy. Se você usar XStreamingElement para passar
saída, então você pode criar uma transformação de arquivo de texto para XML que usa uma quantidade mínima de
memória, independentemente do tamanho do arquivo de texto de origem.
Há algumas restrições em relação às transformações de streaming. Uma transformação de streaming é aplicada
melhor em situações onde você pode processar o arquivo inteiro uma vez, e se você pode processar linhas na
ordem que ocorrem no documento de origem. Se você precisa processar o arquivo mais de uma vez, ou se você
precisa classificar as linhas antes que você possa processar as, você perderá muitos benefícios de usar uma técnica
de streaming.
Exemplo
O seguinte arquivo de texto, People.txt, é a fonte para esse exemplo.
#This is a comment
1,Tai,Yee,Writer
2,Nikolay,Grachev,Programmer
3,David,Wright,Inventor
O código a seguir contém um método de extensão que passa as linhas do arquivo de texto em uma forma adiada.
public static class StreamReaderSequence
{
public static IEnumerable<string> Lines(this StreamReader source)
{
if (source == null)
throw new ArgumentNullException(nameof(source));
string line;
while ((line = source.ReadLine()) != null)
{
yield return line;
}
}
}
class Program
{
static void Main(string[] args)
{
var sr = new StreamReader("People.txt");
var xmlTree = new XStreamingElement("Root",
from line in sr.Lines()
let items = line.Split(',')
where !line.StartsWith("#")
select new XElement("Person",
new XAttribute("ID", items[0]),
new XElement("First", items[1]),
new XElement("Last", items[2]),
new XElement("Occupation", items[3])
)
);
Console.WriteLine(xmlTree);
sr.Close();
}
}
<Root>
<Person ID="1">
<First>Tai</First>
<Last>Yee</Last>
<Occupation>Writer</Occupation>
</Person>
<Person ID="2">
<First>Nikolay</First>
<Last>Grachev</Last>
<Occupation>Programmer</Occupation>
</Person>
<Person ID="3">
<First>David</First>
<Last>Wright</Last>
<Occupation>Inventor</Occupation>
</Person>
</Root>
Consulte também
XStreamingElement
Como: Listar todos os nós em uma árvore (C#)
23/10/2019 • 5 minutes to read • Edit Online
Às vezes é útil listar todos os nós em uma árvore. Isso pode ser útil para saber exatamente como um método ou
propriedade afeta a árvore. Uma abordagem para listar todos os nós em um formato textual é gerar uma expressão
XPath que identifica exata e especificamente qualquer nó na árvore.
Não é particularmente útil executar expressões XPath usando LINQ to XML. As expressões XPath têm um
desempenho pior que consultas LINQ to XML e as consultas LINQ to XML são muito mais avançadas. No entanto,
como uma maneira de identificar nós na árvore XML, o XPath funciona bem.
Exemplo
Este exemplo mostra uma função chamada GetXPath que produz uma expressão XPath específica para qualquer
nó na árvore XML. Ela gera expressões XPath apropriadas mesmo quando os nós estão em um namespace. As
expressões XPath são geradas usando prefixos de namespace.
O exemplo cria uma árvore XML pequena que contém um exemplo de vários tipos de nós. Ela, em seguida, itera
pelos nós descendentes e imprime a expressão XPath para cada nó.
Você observará que a declaração XML não é um nó na árvore.
Veja a seguir um arquivo XML que contém vários tipos de nós:
Veja a seguir a lista de nós na árvore XML anterior, expressa como expressões XPath:
/processing-instruction()
/Root
/Root/@AttName
/Root/@xmlns:aw
/Root/comment()
/Root/Child[1]
/Root/Child[1]/text()
/Root/Child[2]
/Root/Child[2]/text()
/Root/ChildWithMixedContent
/Root/ChildWithMixedContent/text()[1]
/Root/ChildWithMixedContent/b
/Root/ChildWithMixedContent/b/text()
/Root/ChildWithMixedContent/text()[2]
/Root/aw:ElementInNamespace
/Root/aw:ElementInNamespace/aw:ChildInNamespace
class Program
{
static void Main(string[] args)
{
XNamespace aw = "http://www.adventure-works.com";
XDocument doc = new XDocument(
new XDeclaration("1.0", "utf-8", "yes"),
new XProcessingInstruction("target", "data"),
new XElement("Root",
new XAttribute("AttName", "An Attribute"),
new XAttribute(XNamespace.Xmlns + "aw", aw.ToString()),
new XComment("This is a comment"),
new XElement("Child",
new XText("Text")
),
new XElement("Child",
new XText("Other Text")
),
new XElement("ChildWithMixedContent",
new XText("text"),
new XElement("b", "BoldText"),
new XText("otherText")
),
new XElement(aw + "ElementInNamespace",
new XElement(aw + "ChildInNamespace")
)
)
);
doc.Save("Test.xml");
Console.WriteLine(File.ReadAllText("Test.xml"));
Console.WriteLine("------");
foreach (XObject obj in doc.DescendantNodes())
{
Console.WriteLine(obj.GetXPath());
XElement el = obj as XElement;
if (el != null)
foreach (XAttribute at in el.Attributes())
Console.WriteLine(at.GetXPath());
}
}
}
Este tópico apresenta um exemplo que abre um documento do Office Open XML, e retorna uma coleção de todos
os parágrafos no documento.
Para saber mais sobre o Office Open XML, confira SDK do Open XML e www.ericwhite.com.
Exemplo
Este exemplo abre um pacote do Office Open XML, usa os relacionamentos de pacote em Open XML para localizar
o documento e partes de estilo. Consulta no documento, projetando uma coleção de um tipo anônimo que contém
o nó de XElement de parágrafo, o nome do estilo de cada parágrafo, e o texto de cada parágrafo.
O exemplo usa um método de extensão chamado StringConcatenate , que é fornecido no exemplo também.
Para um tutorial detalhado que explica como este exemplo funciona, consulte Transformações funcionais puras de
XML (C#).
Este exemplo usa as classes encontradas no assembly WindowsBase. Ele usa tipos no namespace
System.IO.Packaging.
class Program
{
public static string ParagraphText(XElement e)
{
XNamespace w = e.Name.Namespace;
return e
.Elements(w + "r")
.Elements(w + "t")
.StringConcatenate(element => (string)element);
}
string defaultStyle =
(string)(
from style in styleDoc.Root.Elements(w + "style")
where (string)style.Attribute(w + "type") == "paragraph" &&
(string)style.Attribute(w + "default") == "1"
select style
).First().Attribute(w + "styleId");
).First().Attribute(w + "styleId");
Quando executado com o documento Open XML de exemplo descrito em Criando o documento do Office Open
XML de origem (C#), este exemplo produz a seguinte saída:
Este tópico apresenta um exemplo que abre um documento do Office Open XML, modifica-o e salva-o.
Para saber mais sobre o Office Open XML, confira SDK do Open XML e www.ericwhite.com.
Exemplo
Este exemplo localiza o primeiro elemento de parágrafo no documento. Ele recupera o texto do parágrafo e, em
seguida, exclui todas as execuções do texto no parágrafo. Ele cria uma nova execução de texto que consiste no texto
do primeiro parágrafo que foi convertido para maiúsculas. Ele em seguida serializa o XML modificado no pacote
Open XML e fecha-o.
Este exemplo usa as classes encontradas no assembly WindowsBase. Ele usa tipos no namespace
System.IO.Packaging.
class Program
{
public static string ParagraphText(XElement e)
{
{
XNamespace w = e.Name.Namespace;
return e
.Elements(w + "r")
.Elements(w + "t")
.StringConcatenate(element => (string)element);
}
if (styleRelation != null)
{
Uri styleUri = PackUriHelper.ResolvePartUri(documentUri, styleRelation.TargetUri);
stylePart = wdPackage.GetPart(styleUri);
paraNode.Add(
new XElement(w + "r",
new XElement(w + "t", paraText.ToUpper())
)
);
Se você abrir o SampleDoc.docx depois de executar este programa, verá que este programa converteu o primeiro
parágrafo no documento para maiúsculas.
Quando executado com o documento Open XML de exemplo descrito em Criando o documento do Office Open
XML de origem (C#), este exemplo produz a seguinte saída:
Um aplicativo comum e útil das árvores XML é como um armazenamento de dados hierárquica de nome/valor.
Você pode preencher uma árvore XML com dados hierárquicos, e consultar-la em seguida, transformar-la e, se
necessário, serializar-la. Neste cenário de uso, muitas de semântica específica XML, como namespaces e
comportamento de espaço em branco, não são importantes. Em vez disso, você estiver usando a árvore XML como
um pequeno, na memória, base de dados hierárquica de usuário único.
Exemplo
O exemplo a seguir preenche uma árvore XML do sistema de arquivos local usando a recursão. Consulta na
árvore, calculando o total de tamanhos de todos os arquivos na árvore.
class Program
{
static XElement CreateFileSystemXmlTree(string source)
{
DirectoryInfo di = new DirectoryInfo(source);
return new XElement("Dir",
new XAttribute("Name", di.Name),
from d in Directory.GetDirectories(source)
select CreateFileSystemXmlTree(d),
from fi in di.GetFiles()
select new XElement("File",
new XElement("Name", fi.Name),
new XElement("Length", fi.Length)
)
);
}
O XPath e o LINQ to XML fornecem alguma funcionalidade semelhante. Ambos podem ser usados para ver uma
árvore XML, retornando resultados como uma coleção de elementos, uma coleção de atributos, uma coleção de
nós, ou o valor de um elemento ou de um atributo. No entanto, também há algumas diferenças.
Ordenação de resultado
O XPath 1,0 estados de recomendação que uma coleção que é o resultado de avaliar uma expressão XPath é não
ordenada.
Entretanto, ao fazer iterações por uma coleção retornada por um método do eixo XPath do LINQ to XML, os nós na
coleção são retornados na ordem do documento. Este é o caso mesmo quando acessando os eixos XPath onde os
predicados são expressos em termos de ordem inversa de documento, como preceding e preceding-sibling .
Por outro lado, a maioria dos eixos do LINQ to XML retorna coleções na ordem de documento, mas por dois deles,
Ancestors e AncestorsAndSelf, retornam coleções na ordem inversa do documento. A tabela a seguir enumera os
eixos, e indica a ordem de coleção para cada:
Predicados posicionais
Em uma expressão XPath, os predicados posicionais são expressos em termos de pedido de documento para
muitos eixos, mas expressos na ordem inversa de documento para os eixos invertidos, que são preceding ,
preceding-sibling , ancestor , e ancestor-or-self . Por exemplo, a expressão XPath preceding-sibling::*[1]
retorna imediatamente antes o irmão. Este é o caso mesmo que o conjunto de resultados final é apresentado na
ordem de documento.
Por outro lado, todos os predicados posicionais em LINQ to XML sempre são expressos em termos de pedido do
eixo. Por exemplo, anElement.ElementsBeforeSelf().ElementAt(0) retorna o primeiro elemento filho do pai do
elemento consultado, não irmão anterior imediato. Outro exemplo: anElement.Ancestors().ElementAt(0) retorna o
elemento pai.
Se você desejar localizar imediatamente o elemento precedente no LINQ to XML, escreva a expressão a seguir:
ElementsBeforeSelf().Last()
ElementsBeforeSelf().Last()
Diferenças de desempenho
As consultas XPath que usam a funcionalidade XPath no LINQ to XML não serão tão bem executadas como as
consultas do LINQ to XML.
Comparação de composição
A composição de uma consulta do LINQ to XML é um pouco paralela à composição de uma expressão XPath,
embora muito diferente na sintaxe.
Por exemplo, se você tiver um elemento em uma variável chamada customers , e você desejar localizar um
elemento de neto chamado CompanyName em todos os elementos filho chamados Customer , escreva uma expressão
XPath como segue:
customers.XPathSelectElements("./Customer/CompanyName")
customers.XPathSelectElements("./Customer/CompanyName")
customers.Elements("Customer").Elements("CompanyName")
customers.Elements("Customer").Elements("CompanyName")
ou
XElement.Attributes
ou
XContainer.DescendantNodes
descendente-ou-auto XElement.DescendantsAndSelf
ou
XElement.DescendantNodesAndSelf
seguir-irmão XNode.ElementsAfterSelf
ou
XNode.NodesAfterSelf
acima-irmão XNode.ElementsBeforeSelf
ou
XNode.NodesBeforeSelf
Este tópico compara o eixo de elemento filho XPath com o método LINQ to XML Element.
A expressão XPath é DeliveryNotes .
Exemplo
Este exemplo localiza o elemento filho DeliveryNotes .
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: vários pedidos de compra (LINQ to
XML ).
// XPath expression
XElement el2 = po.XPathSelectElement("DeliveryNotes");
// same as "child::DeliveryNotes"
// same as "./DeliveryNotes"
if (el1 == el2)
Console.WriteLine("Results are identical");
else
Console.WriteLine("Results differ");
Console.WriteLine(el1);
Este tópico compara o eixo de elementos filho XPath com o eixo LINQ to XML Elements.
A expressão XPath é: ./*
Exemplo
Este exemplo localiza os elementos filho do elemento Address .
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: vários pedidos de compra (LINQ to
XML ).
// XPath expression
IEnumerable<XElement> list2 = po.XPathSelectElements("./*");
Este tópico mostra como obter o elemento raiz com XPath e LINQ to XML.
A expressão XPath é:
/PurchaseOrders
Exemplo
Este exemplo localiza o elemento raiz.
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: Várias ordens de compra (LINQ to XML ).
XDocument po = XDocument.Load("PurchaseOrders.xml");
// XPath expression
XElement el2 = po.XPathSelectElement("/PurchaseOrders");
if (el1 == el2)
Console.WriteLine("Results are identical");
else
Console.WriteLine("Results differ");
Console.WriteLine(el1.Name);
Este tópico mostra como obter os elementos descendentes com um nome específico.
A expressão XPath é //Name .
Exemplo
Este exemplo localiza os descendentes chamados Name .
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: vários pedidos de compra (LINQ to
XML ).
XDocument po = XDocument.Load("PurchaseOrders.xml");
// XPath expression
IEnumerable<XElement> list2 = po.XPathSelectElements("//Name");
Este tópico mostra como obter os elementos descendentes com um nome especificado e, com um atributo com um
valor especificado.
A expressão XPath é:
.//Address[@Type='Shipping']
Exemplo
Este exemplo localiza os elementos dos descendentes com o nome de Address , e com um atributo de Type com
um valor de “enviar”.
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: vários pedidos de compra (LINQ to
XML ).
XDocument po = XDocument.Load("PurchaseOrders.xml");
// XPath expression
IEnumerable<XElement> list2 = po.XPathSelectElements(".//Address[@Type='Shipping']");
Este tópico mostra como obter um elemento que seleciona em um atributo que é chamada pelo valor de outro
elemento.
A expressão XPath é:
.//Customer[@CustomerID=/Root/Orders/Order[12]/CustomerID]
Exemplo
Este exemplo localiza o 12o elemento de Order , e localiza no cliente para essa ordem.
Observe que a indexação em uma lista no .NET é com base em “zero”. A indexação em uma coleção de nós em um
predicado XPath “é um” com base. Este exemplo reflete essa diferença.
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: clientes e pedidos (LINQ to XML ).
XDocument co = XDocument.Load("CustomersOrders.xml");
// XPath expression
XElement customer3 = co.XPathSelectElement(
".//Customer[@CustomerID=/Root/Orders/Order[12]/CustomerID]");
As expressões XPath pode localizar nós em um namespace específico. Prefixos de namespace do uso de expressões
XPath para especificar namespaces. Para analisar uma expressão XPath que contém prefixos de namespace, você
deve passar um objeto para os métodos XPath que implementa IXmlNamespaceResolver. Este exemplo usa
XmlNamespaceManager.
A expressão XPath é:
./aw:*
Exemplo
O exemplo a seguir lê uma árvore XML que contém dois namespaces. Usa XmlReader para ler o documento XML.
Então obtém XmlNameTable de XmlReader, e XmlNamespaceManager de XmlNameTable. Usa
XmlNamespaceManager ao selecionar elementos.
Este tópico compara o eixo preceding-sibling de XPath ao eixo XNode.ElementsBeforeSelf filho do LINQ to XML.
A expressão XPath é:
preceding-sibling::*
Exemplo
O exemplo a seguir localiza o elemento de FullAddress , e então recuperar os elementos anteriores usando o eixo
de preceding-sibling .
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: clientes e pedidos (LINQ to XML ).
XElement co = XElement.Load("CustomersOrders.xml");
// XPath expression
IEnumerable<XElement> list2 = add.XPathSelectElements("preceding-sibling::*");
Este tópico mostra como obter os elementos descendentes de um elemento filho com um nome específico.
A expressão XPath é:
./Paragraph//Text/text()
Exemplo
Este exemplo simula problemas de extrair o texto de uma representação de um documento de processamento de
texto. Seleciona primeiro todos os elementos de Paragraph , e então seleciona todos os elementos descendentes de
Text de cada elemento de Paragraph . Isso não selecionar os elementos de Text descendente do elemento de
Comment .
XElement root = XElement.Parse(
@"<Root>
<Paragraph>
<Text>This is the start of</Text>
</Paragraph>
<Comment>
<Text>This comment is not part of the paragraph text.</Text>
</Comment>
<Paragraph>
<Annotation Emphasis='true'>
<Text> a sentence.</Text>
</Annotation>
</Paragraph>
<Paragraph>
<Text> This is a second sentence.</Text>
</Paragraph>
</Root>");
// XPath expression
string str2 =
((IEnumerable)root.XPathEvaluate("./Paragraph//Text/text()"))
.Cast<XText>()
.Select(s => s.Value)
.Aggregate(
new StringBuilder(),
(s, i) => s.Append(i),
s => s.ToString()
);
if (str1 == str2)
Console.WriteLine("Results are identical");
else
Console.WriteLine("Results differ");
Console.WriteLine(str2);
O XPath permite que você localize a união de resultados de dois caminhos de local XPath.
A expressão XPath é:
//Category|//Price
Você pode obter os mesmos resultados usando o operador padrão de consulta de Concat .
Exemplo
Este exemplo localiza os elementos de Category e todos os elementos de Price , e os concatena em uma única
coleção. Observe que a consulta de LINQ to XML chama InDocumentOrder para ordenar os resultados. Os
resultados da avaliação de expressão XPath são também em ordem do documento.
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: dados numéricos (LINQ to XML ).
// XPath expression
IEnumerable<XElement> list2 = data.XPathSelectElements("//Category|//Price");
Você pode desejar localizar todos os seus irmãos de um nó que têm um nome específico. A coleção resultante pode
incluir o nó de contexto se o nó de contexto também tem o nome específico.
A expressão XPath é:
../Book
Exemplo
Este exemplo localiza primeiro um elemento de Book , e localiza em todos os elementos irmãos nomeados Book .
A coleção resultante inclui o nó de contexto.
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: livros (LINQ to XML ).
XElement book =
books
.Root
.Elements("Book")
.Skip(1)
.First();
// XPath expression
IEnumerable<XElement> list2 = book.XPathSelectElements("../Book");
Este tópico mostra como navegar para o elemento pai e localizar um atributo deles.
A expressão XPath é:
../@id
Exemplo
Este exemplo localiza primeiro um elemento de Author . Localiza no atributo de id de elemento pai.
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: livros (LINQ to XML ).
XElement author =
books
.Root
.Element("Book")
.Element("Author");
// XPath expression
XAttribute att2 = ((IEnumerable)author.XPathEvaluate("../@id")).Cast<XAttribute>().First();
if (att1 == att2)
Console.WriteLine("Results are identical");
else
Console.WriteLine("Results differ");
Console.WriteLine(att1);
Este tópico mostra como localizar todos os atributos de seus irmãos o nó de contexto. Somente os atributos com
um nome específico são retornados na coleção.
A expressão XPath é:
../Book/@id
Exemplo
Este exemplo localiza primeiro um elemento de Book , e localiza em todos os elementos irmãos nomeados Book ,e
localiza em todos os atributos nomeados id . O resultado é uma coleção de atributos.
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: livros (LINQ to XML ).
XElement book =
books
.Root
.Element("Book");
// XPath expression
IEnumerable<XAttribute> list2 =
((IEnumerable)book.XPathEvaluate("../Book/@id")).Cast<XAttribute>();
Às vezes você deseja localizar todos os elementos que têm um atributo específico. Você não está preocupado com
o conteúdo do atributo. Você quer selecionar com base na existência do atributo.
A expressão XPath é:
./*[@Select]
Exemplo
O código a seguir seleciona apenas os elementos que têm o atributo Select .
// XPath expression
IEnumerable<XElement> list2 =
((IEnumerable)doc.XPathEvaluate("./*[@Select]")).Cast<XElement>();
Às vezes você deseja localizar elementos com base em sua posição. Você pode desejar localizar o segundo
elemento, ou você talvez queira localizar o terceiro através do quinto elemento.
A expressão XPath é:
Test[position() >= 2 and position() <= 4]
Há duas abordagens para escrever esta consulta de LINQ to XML de uma maneira lazy. Você pode usar os
operadores de Skip e de Take , ou você pode usar a sobrecarga de Where que usa um índice. Quando você usa a
sobrecarga de Where , você usa uma expressão lambda que leva dois argumentos. O exemplo a seguir mostra os
dois métodos de selecione a posição de functionamento base.
Exemplo
Este exemplo localiza o segundo a quarto elemento de Test . O resultado é uma coleção de elementos.
Este exemplo usa o seguinte documento XML: Arquivo XML de exemplo: configuração de teste (LINQ to XML ).
// XPath expression
IEnumerable<XElement> list3 =
testCfg.XPathSelectElements("Test[position() >= 2 and position() <= 4]");
Às vezes você deseja encontrar o irmão anterior imediato a um nó. Devido a diferença na semântica de predicados
posicionais para os eixos anterior irmãos no XPath ao contrário de LINQ to XML, essa é uma das comparações
mais interessantes.
Exemplo
Neste exemplo, a consulta LINQ to XML usa o operador Last para encontrar o último nó na coleção retornada por
ElementsBeforeSelf. Por outro lado, a expressão XPath usa um predicado com um valor de 1 para localizar o
elemento imediatamente anterior.
// XPath expression
XElement el2 =
((IEnumerable)child4
.XPathEvaluate("preceding-sibling::*[1]"))
.Cast<XElement>()
.First();
if (el1 == el2)
Console.WriteLine("Results are identical");
else
Console.WriteLine("Results differ");
Console.WriteLine(el1);
Nesta seção
TÓPICO DESCRIÇÃO
Programação funcional versus Programação obrigatória (C#) Compara e contrasta programação funcional à programação
(procedural) imperativa mais tradicional.
Refatoração em funções puras (C#) Apresenta funções puras, e mostra exemplos de funções e
puras e impuros.
Aplicabilidade da transformação funcional (C#) Descreve cenários típicos para transformações funcionais.
IMPORTANT
No restante deste tutorial, o termo “function” pura é usado em um sentido geral indicar uma abordagem de programação, e
não em um recurso de linguagem específica.
Observe que as funções puras devem ser implementadas como métodos no C#.
Além disso, você não deve confundir funções puras com os métodos puros virtuais em C++. O último indica que a classe
recipiente é abstrato e que nenhum corpo do método é fornecido.
Programação funcional
A programação funcional é uma abordagem de programação que dá suporte diretamente à transformação
funcional pura.
Historicamente, as linguagens de programação funcionais de uso geral, como o ML, Scheme, Haskell e F#, têm
sido, principalmente, de interesse da comunidade acadêmica. Embora sempre foi possível escrever transformações
funcionais puras no C#, a dificuldade em fazê-lo não tornou essa opção atrativa à maioria dos programadores. Nas
versões recentes do C#, no entanto, os novos constructos de linguagem como expressões lambda e inferência de
tipos tornaram a programação funcional muito mais fácil e mais produtiva.
Para obter mais informações sobre programação funcional, consulte Programação funcional versus Programação
obrigatória (C#).
Linguagens específicas do domínio FP
Embora as linguagens de programação e gerais não sejam amplamente adotadas, as linguagens de programação e
específicas do domínio específicas tinham melhor êxito. Por exemplo, folhas de estilos em cascata (CSS ) são
usadas para determinar a aparência de muitas páginas da Web, e as folhas de estilos extensíveis de transformações
de language (XSLT) são usadas amplamente na manipulação de dados XML. Para obter mais informações sobre
XSLT, consulte Transformações XSLT.
Terminologia
A tabela a seguir define os termos relacionados às transformações funcionais.
função de ordem superior (primeira classe)
Uma função que pode ser tratado como um objeto através de programação. Por exemplo, uma função de pedido
superior pode ser passada para ou retornado de outras funções. No C#, delegados e expressões lambda são
recursos de linguagem que dão suporte a funções de ordem superior. Para gravar uma função de pedido superior,
você declara um ou mais argumentos para tomar representantes, e você frequentemente usa expressões lambda
para chamá-lo. Muitos dos operadores de consulta padrão são funções de pedido superior.
Para obter mais informações, consulte Visão geral de operadores de consulta padrão (C#).
expressão lambda
Essencialmente, uma função anônimo embutido que pode ser usada em que um tipo delegate é esperada. Esta é
uma definição simplificada de expressões lambda, mas é suficiente para fins deste tutorial.
Para obter mais informações, consulte Expressões lambda.
coleção
Um conjunto estruturada de dados, geralmente um tipo de uniforme. Para ser compatível com LINQ, uma coleção
deve implementar a interface de IEnumerable ou a interface de IQueryable (ou uma de suas contrapartes
genéricos, de IEnumerator<T> ou de IQueryable<T>).
tupla (tipos anônimos)
Um conceito matemático, um tuple é uma sequência finito rotuladas de objetos, cada um de um tipo específico.
Um tuple também é conhecido como uma lista ordenada. Tipos anônimos são uma implementação de linguagem
desse conceito, que permitem que um tipo sem nome da classe ser declarados e um objeto do tipo a ser
instanciada ao mesmo tempo.
Para obter mais informações, consulte Tipos anônimos.
inferência de tipos (digitação implícita)
A capacidade de um compilador de determinar o tipo de uma variável na ausência de uma declaração de tipo
explícita.
Para obter mais informações, consulte Variáveis locais de tipo implícito.
execução adiada e avaliação lenta
O atraso de avaliação de uma expressão até que o valor resolvido é realmente necessário. A execução adiada é
suportado em coleções.
Para obter mais informações, consulte Introdução a consultas LINQ (C#) e Execução adiada e avaliação lenta em
LINQ to XML (C#).
Esses recursos de idioma serão usados em exemplos de código em todo esta seção.
Consulte também
Introdução às transformações funcionais puras (C#)
Programação funcional versus Programação obrigatória (C#)
Programação com funcional. Programação
obrigatória (C#)
23/10/2019 • 8 minutes to read • Edit Online
Este tópico compara e contrasta programação funcional com programação (procedural) imperativa mais
tradicional.
Foco do programador Como executar tarefas (algoritmos) e Informações que é desejada e que
como controlar alterações no estado. transformações são necessárias.
Controle de fluxo primária Loop, condições, e chamadas de função Chamadas de função, incluindo a
(método). recursão.
Unidade principal de manipulação Instâncias das classes ou estruturas. Funções como objetos de primeira
classe e coleções de dados.
Embora a maioria das linguagens sejam criados para oferecer suporte a um paradigma específico de
programação, vários idiomas gerais é flexível o suficiente para suportar várias paradigma. Por exemplo, a maioria
das linguagens que contêm ponteiros de função podem ser usados para oferecer suporte digna de crédito
programação funcional. Além disso, o C# inclui extensões de linguagem explícitas para dar suporte à
programação funcional, incluindo expressões lambda e inferência de tipos. A tecnologia LINQ é um formulário de
programação declarativa, funcional.
Consulte também
Introdução às transformações funcionais puras (C#)
Transformações XSLT
Refatoração em funções puras (C#)
Refatoração em funções puras (C#)
23/10/2019 • 6 minutes to read • Edit Online
Um aspecto importante de transformações e puras é aprender como o código do refatorar usando funções puras.
NOTE
A nomenclatura comuns em programação funcional é que você refatora programa usando funções puras. No Visual Basic e
C++, isso alinha com o uso das funções em linguagens respectivos. No entanto, em C#, as funções são chamadas métodos.
Para fins desta discussão, uma função pura é implementada como um método em C#.
Conforme observado anteriormente nesta seção, uma função pura tem duas características úteis:
Não tem efeito colateral. A função não altera quaisquer variáveis ou os dados de qualquer tipo fora da
função.
É consistente. Dado o mesmo conjunto de dados de entrada, sempre retornará o mesmo valor de saída.
Uma maneira de fazer a transição para programação funcional é o código existente do refatorar para eliminar
efeitos colaterais desnecessários e dependências externas. Dessa maneira, você pode criar versões puras de
função do código existente.
Este tópico descreve o que é uma função pura e o que não é. O Tutorial: Manipulando o conteúdo em um
documento WordprocessingML (C#) mostra como manipular um documento WordprocessingML e inclui dois
exemplos de como fazer a refatoração usando uma função pura.
Observe que é irrelevante se os dados modificados têm acesso a public ou private , ou são um membro de
static ou um membro de instância. Uma função pura não altera quaisquer dados fora da função.
Esta versão do programa gerenciar as mesmas saída que a primeira versão, porque a função de HyphenatedConcat
alterou o valor (estado) do primeiro parâmetro chamar a função de membro de Append . Observe que essa
mudança ocorre independentemente do fato de que HyphenatedConcat usa passar o parâmetro de atendimento-
por- valor.
IMPORTANT
Para tipos de referência, se você passa um parâmetro por valor, ele resulta em uma cópia de referência a um objeto que está
sendo passado. Esta cópia ainda está associada com os mesmos dados de instância que a referência original (até que a
variável de referência é atribuído a um novo objeto). a Atendimento-por- referência não necessariamente é necessária para
uma função modifique um parâmetro.
Função pura
Esta próxima versão do programa mostra como implementar a função HyphenatedConcat como uma função pura.
class Program
{
public static string HyphenatedConcat(string s, string appendStr)
{
return (s + '-' + appendStr);
}
Além disso, esta versão gerencia a mesma linha de saída: StringOne-StringTwo . Observe que para manter o valor
concatenado, ele é armazenado em s2 variável intermediária.
Uma abordagem que pode ser muito útil é escrever as funções que são localmente impuros (isto é, declare e
alteram variáveis locais) mas é global pura. Tais funções têm muitas das características desejáveis de
composability, mas impedem alguns de linguagem e mais complicados de programação, como ter que usar a
recursão quando um loop simples realizaria a mesma coisa.
Consulte também
Introdução às transformações funcionais puras (C#)
Programação funcional versus Programação obrigatória (C#)
Aplicabilidade da transformação funcional (C#)
23/10/2019 • 3 minutes to read • Edit Online
Consulte também
Introdução às transformações funcionais puras (C#)
Transformação funcional de XML (C#)
Refatoração em funções puras (C#)
Transformação funcional de XML (C#)
23/10/2019 • 3 minutes to read • Edit Online
Este tópico descreve a abordagem funcional pura de transformação aos documentos XML de alteração, e ele
contrastes com uma abordagem procedural.
Consulte também
Introdução às transformações funcionais puras (C#)
Tutorial: manipulando conteúdo em um documento WordprocessingML (C#)
LINQ to XML versus Outras Tecnologias XML
Execução adiada e avaliação lenta em LINQ to XML
(C#)
23/10/2019 • 3 minutes to read • Edit Online
As operações de consulta e eixo são geralmente implementadas para usar a execução adiada. Este tópico explica os
requisitos e as vantagens da execução adiada, além de algumas considerações sobre a implementação.
Execução Adiada
A execução adiada significa que a avaliação de uma expressão será atrasada até que seu valor realizado seja
realmente necessário. A execução adiada pode aumentar muito o desempenho quando você precisa manipular
grandes coleções de dados, especialmente em programas que contêm uma série de consultas ou manipulações
encadeadas. Na melhor das hipóteses, a execução adiada permite apenas uma única iteração pela coleção de
origem.
As tecnologias LINQ usam de modo intenso a execução adiada em ambos os membros das classes System.Linq
centrais e nos métodos de extensão nos diversos namespaces LINQ, como System.Xml.Linq.Extensions.
A execução adiada tem suporte direto na linguagem C# pela palavra-chave yield (na forma da instrução
yield-return ) quando usada dentro de um bloco de iterador. Tal iterador deve retornar uma coleção do tipo
IEnumerator ou IEnumerator<T> (ou um tipo derivado).
Próximas etapas
O próximo tópico deste tutorial ilustra a execução adiada:
Exemplo de execução adiada (C#)
Consulte também
Tutorial: Encadeando consultas (C#)
Conceitos e terminologia (transformação funcional) (C#)
Operações de agregação (C#)
yield
Exemplo de execução adiada (C#)
23/10/2019 • 2 minutes to read • Edit Online
Este tópico mostra como execução adiada e a avaliação lazy afetam a execução das consultas LINQ to XML.
Exemplo
O exemplo a seguir mostra a ordem de execução para usar um método de extensão que use a execução adiada. O
exemplo declara uma matriz de três cadeias de caracteres. Em itera através da coleção retornada por
ConvertCollectionToUpperCase .
class Program
{
static void Main(string[] args)
{
string[] stringArray = { "abc", "def", "ghi" };
Observe que para iterar através da coleção retornada por ConvertCollectionToUpperCase , cada item é recuperado de
matriz de cadeias de caracteres de origem e convertido para maiúsculas antes que o próximo item é recuperado de
matriz de cadeias de caracteres de origem.
Você pode ver que a matriz inteira de cadeias de caracteres não é convertida para maiúsculas antes que cada item
na coleção retornada é processado no loop de foreach em Main .
O próximo tópico neste tutorial mostra o encadeamento consultas em conjunto:
Exemplo de encadeamento de consultas (C#)
Consulte também
Tutorial: Encadeando consultas (C#)
Exemplo de encadeamento de consultas (C#)
23/10/2019 • 2 minutes to read • Edit Online
Este exemplo compila no exemplo anterior e mostra o que acontece quando você encadeia duas consultas que
usam execução adiada e avaliação lenta.
Exemplo
Nesse exemplo, um outro método de extensão é apresentado, AppendString , que acrescenta uma cadeia de
caracteres especificada em cada cadeia de caracteres na coleção de origem, e produz nas novas cadeias de
caracteres.
class Program
{
static void Main(string[] args)
{
string[] stringArray = { "abc", "def", "ghi" };
IEnumerable<string> q1 =
from s in stringArray.ConvertCollectionToUpperCase()
select s;
IEnumerable<string> q2 =
from s in q1.AppendString("!!!")
select s;
Nesse exemplo, você pode ver que cada método de extensão opera um de cada vez para cada item na coleção de
origem.
O que deve ser claro desse exemplo é o mesmo que nós encadeemos juntos consultas que rendem coleções,
qualquer coleção intermediária é materializada. Em vez disso, cada item é passado de um método lenta a seguir.
Isso resulta em uma pegada muito menor de memória do que uma abordagem que recebe primeiro uma matriz de
cadeias de caracteres, então cria uma segunda matriz de cadeias de caracteres que foram convertidos para
maiúsculas, e criar finalmente uma terceira matriz de cadeias de caracteres onde cada cadeia de caracteres tem os
pontos de exclamação acrescentados a ele.
O próximo tópico neste tutorial mostra o materialization intermediário:
Materialização intermediária (C#)
Consulte também
Tutorial: Encadeando consultas (C#)
Materialização intermediária (C#)
23/10/2019 • 2 minutes to read • Edit Online
Se você não for cauteloso, em algumas situações dràstica você pode alterar a memória e o perfil de desempenho
do seu aplicativo causando o materialization prematuro das coleções nas suas consultas. Alguns operadores de
consulta padrão faz com que o materialization de sua coleção de origem antes de produzir um único elemento. Por
exemplo, Enumerable.OrderBy primeiro itera através da coleção inteira de origem, então classe todos os itens, e
então produz basicamente o primeiro item. Isso significa que é grande obter o primeiro item de uma coleção
ordenada; cada item não for depois disso caro. Isso faz sentido: Seria impossível para esse operador de consulta
fazer de outra maneira.
Exemplo
Este exemplo altera o exemplo anterior. O método de AppendString chama ToList antes de fazer iterações pela
origem. Isso faz com que o materialization.
public static class LocalExtensions
{
public static IEnumerable<string>
ConvertCollectionToUpperCase(this IEnumerable<string> source)
{
foreach (string str in source)
{
Console.WriteLine("ToUpper: source >{0}<", str);
yield return str.ToUpper();
}
}
class Program
{
static void Main(string[] args)
{
string[] stringArray = { "abc", "def", "ghi" };
IEnumerable<string> q1 =
from s in stringArray.ConvertCollectionToUpperCase()
select s;
IEnumerable<string> q2 =
from s in q1.AppendString("!!!")
select s;
Nesse exemplo, você pode ver que a chamada a ToList faz com que AppendString enumerar sua fonte inteiro antes
de como o primeiro item. Se a origem foi uma matriz grande, essa alteraria significativamente o perfil de memória
do aplicativo.
Os operadores de consulta padrão podem também ser encadeados juntos. O final deste tópico ilustra este tutorial.
Encadeando operadores de consulta padrão juntos (C#)
Consulte também
Tutorial: Encadeando consultas (C#)
Encadeando operadores de consulta padrão juntos
(C#)
23/10/2019 • 2 minutes to read • Edit Online
Exemplo
Nesse exemplo, o método de Where é chamado antes de chamar ConvertCollectionToUpperCase . O método de
Where opera quase exatamente a mesma maneira que os métodos lentos usados em exemplos anteriores neste
tutorial, ConvertCollectionToUpperCase e AppendString .
Uma diferença é que nesse caso, o método de Where itera através da coleção fonte, determina que o primeiro item
não passa o predicado e em seguida, o próximo item, que passa. Produz no segundo item.
No entanto, a ideia básica é a mesma: As coleções intermediárias não são materializadas a menos que isso seja
necessário.
Quando as expressões de consulta são usadas, são convertidas para chamadas aos operadores de consulta padrão,
e os mesmos princípios se aplicam.
Todos os exemplos nesta seção que estão vendo documentos do Office Open XML usam o mesmo princípio. A
execução adiada e a avaliação lazy são alguns conceitos fundamentais que você deve compreender para usar
efetivamente LINQ (e LINQ to XML ).
public static class LocalExtensions
{
public static IEnumerable<string>
ConvertCollectionToUpperCase(this IEnumerable<string> source)
{
foreach (string str in source)
{
Console.WriteLine("ToUpper: source >{0}<", str);
yield return str.ToUpper();
}
}
class Program
{
static void Main(string[] args)
{
string[] stringArray = { "abc", "def", "ghi" };
IEnumerable<string> q1 =
from s in stringArray.ConvertCollectionToUpperCase()
where s.CompareTo("D") >= 0
select s;
IEnumerable<string> q2 =
from s in q1.AppendString("!!!")
select s;
As operações de consulta e eixo são geralmente implementadas para usar a execução adiada. Este tópico
explica os requisitos e as vantagens da execução adiada, além de algumas considerações sobre a
implementação.
Execução Adiada
A execução adiada significa que a avaliação de uma expressão será atrasada até que seu valor realizado seja
realmente necessário. A execução adiada pode aumentar muito o desempenho quando você precisa manipular
grandes coleções de dados, especialmente em programas que contêm uma série de consultas ou
manipulações encadeadas. Na melhor das hipóteses, a execução adiada permite apenas uma única iteração
pela coleção de origem.
As tecnologias LINQ usam de modo intenso a execução adiada em ambos os membros das classes
System.Linq centrais e nos métodos de extensão nos diversos namespaces LINQ, como
System.Xml.Linq.Extensions.
A execução adiada tem suporte direto na linguagem C# pela palavra-chave yield (na forma da instrução
yield-return ) quando usada dentro de um bloco de iterador. Tal iterador deve retornar uma coleção do tipo
IEnumerator ou IEnumerator<T> (ou um tipo derivado).
Próximas etapas
O próximo tópico deste tutorial ilustra a execução adiada:
Exemplo de execução adiada (C#)
Consulte também
Tutorial: Encadeando consultas (C#)
Conceitos e terminologia (transformação funcional) (C#)
Operações de agregação (C#)
yield
Exemplo de execução adiada (C#)
23/10/2019 • 2 minutes to read • Edit Online
Este tópico mostra como execução adiada e a avaliação lazy afetam a execução das consultas LINQ to XML.
Exemplo
O exemplo a seguir mostra a ordem de execução para usar um método de extensão que use a execução adiada. O
exemplo declara uma matriz de três cadeias de caracteres. Em itera através da coleção retornada por
ConvertCollectionToUpperCase .
class Program
{
static void Main(string[] args)
{
string[] stringArray = { "abc", "def", "ghi" };
Observe que para iterar através da coleção retornada por ConvertCollectionToUpperCase , cada item é recuperado
de matriz de cadeias de caracteres de origem e convertido para maiúsculas antes que o próximo item é recuperado
de matriz de cadeias de caracteres de origem.
Você pode ver que a matriz inteira de cadeias de caracteres não é convertida para maiúsculas antes que cada item
na coleção retornada é processado no loop de foreach em Main .
O próximo tópico neste tutorial mostra o encadeamento consultas em conjunto:
Exemplo de encadeamento de consultas (C#)
Consulte também
Tutorial: Encadeando consultas (C#)
Exemplo de encadeamento de consultas (C#)
23/10/2019 • 2 minutes to read • Edit Online
Este exemplo compila no exemplo anterior e mostra o que acontece quando você encadeia duas consultas que
usam execução adiada e avaliação lenta.
Exemplo
Nesse exemplo, um outro método de extensão é apresentado, AppendString , que acrescenta uma cadeia de
caracteres especificada em cada cadeia de caracteres na coleção de origem, e produz nas novas cadeias de
caracteres.
class Program
{
static void Main(string[] args)
{
string[] stringArray = { "abc", "def", "ghi" };
IEnumerable<string> q1 =
from s in stringArray.ConvertCollectionToUpperCase()
select s;
IEnumerable<string> q2 =
from s in q1.AppendString("!!!")
select s;
Nesse exemplo, você pode ver que cada método de extensão opera um de cada vez para cada item na coleção de
origem.
O que deve ser claro desse exemplo é o mesmo que nós encadeemos juntos consultas que rendem coleções,
qualquer coleção intermediária é materializada. Em vez disso, cada item é passado de um método lenta a seguir.
Isso resulta em uma pegada muito menor de memória do que uma abordagem que recebe primeiro uma matriz
de cadeias de caracteres, então cria uma segunda matriz de cadeias de caracteres que foram convertidos para
maiúsculas, e criar finalmente uma terceira matriz de cadeias de caracteres onde cada cadeia de caracteres tem os
pontos de exclamação acrescentados a ele.
O próximo tópico neste tutorial mostra o materialization intermediário:
Materialização intermediária (C#)
Consulte também
Tutorial: Encadeando consultas (C#)
Materialização intermediária (C#)
23/10/2019 • 2 minutes to read • Edit Online
Se você não for cauteloso, em algumas situações dràstica você pode alterar a memória e o perfil de desempenho
do seu aplicativo causando o materialization prematuro das coleções nas suas consultas. Alguns operadores de
consulta padrão faz com que o materialization de sua coleção de origem antes de produzir um único elemento. Por
exemplo, Enumerable.OrderBy primeiro itera através da coleção inteira de origem, então classe todos os itens, e
então produz basicamente o primeiro item. Isso significa que é grande obter o primeiro item de uma coleção
ordenada; cada item não for depois disso caro. Isso faz sentido: Seria impossível para esse operador de consulta
fazer de outra maneira.
Exemplo
Este exemplo altera o exemplo anterior. O método de AppendString chama ToList antes de fazer iterações pela
origem. Isso faz com que o materialization.
public static class LocalExtensions
{
public static IEnumerable<string>
ConvertCollectionToUpperCase(this IEnumerable<string> source)
{
foreach (string str in source)
{
Console.WriteLine("ToUpper: source >{0}<", str);
yield return str.ToUpper();
}
}
class Program
{
static void Main(string[] args)
{
string[] stringArray = { "abc", "def", "ghi" };
IEnumerable<string> q1 =
from s in stringArray.ConvertCollectionToUpperCase()
select s;
IEnumerable<string> q2 =
from s in q1.AppendString("!!!")
select s;
Nesse exemplo, você pode ver que a chamada a ToList faz com que AppendString enumerar sua fonte inteiro antes
de como o primeiro item. Se a origem foi uma matriz grande, essa alteraria significativamente o perfil de memória
do aplicativo.
Os operadores de consulta padrão podem também ser encadeados juntos. O final deste tópico ilustra este tutorial.
Encadeando operadores de consulta padrão juntos (C#)
Consulte também
Tutorial: Encadeando consultas (C#)
Encadeando operadores de consulta padrão juntos
(C#)
23/10/2019 • 2 minutes to read • Edit Online
Exemplo
Nesse exemplo, o método de Where é chamado antes de chamar ConvertCollectionToUpperCase . O método de
Where opera quase exatamente a mesma maneira que os métodos lentos usados em exemplos anteriores neste
tutorial, ConvertCollectionToUpperCase e AppendString .
Uma diferença é que nesse caso, o método de Where itera através da coleção fonte, determina que o primeiro item
não passa o predicado e em seguida, o próximo item, que passa. Produz no segundo item.
No entanto, a ideia básica é a mesma: As coleções intermediárias não são materializadas a menos que isso seja
necessário.
Quando as expressões de consulta são usadas, são convertidas para chamadas aos operadores de consulta padrão,
e os mesmos princípios se aplicam.
Todos os exemplos nesta seção que estão vendo documentos do Office Open XML usam o mesmo princípio. A
execução adiada e a avaliação lazy são alguns conceitos fundamentais que você deve compreender para usar
efetivamente LINQ (e LINQ to XML ).
public static class LocalExtensions
{
public static IEnumerable<string>
ConvertCollectionToUpperCase(this IEnumerable<string> source)
{
foreach (string str in source)
{
Console.WriteLine("ToUpper: source >{0}<", str);
yield return str.ToUpper();
}
}
class Program
{
static void Main(string[] args)
{
string[] stringArray = { "abc", "def", "ghi" };
IEnumerable<string> q1 =
from s in stringArray.ConvertCollectionToUpperCase()
where s.CompareTo("D") >= 0
select s;
IEnumerable<string> q2 =
from s in q1.AppendString("!!!")
select s;
Recursos externos
Apresentando os formatos de arquivo do Office (2007) Open XML
Visão geral de WordprocessingML
Anatomia de um arquivo de WordProcessingML
Introdução a WordprocessingML
Office 2003: página de download de esquemas de referência de XML
Consulte também
Tutorial: manipulando conteúdo em um documento WordprocessingML (C#)
Criando o documento do Office Open XML de
origem (C#)
23/10/2019 • 2 minutes to read • Edit Online
Este tópico mostra como criar o documento do Office Open XML WordprocessingML que os outros exemplos
neste tutorial uso. Se você segue essas declarações, a saída corresponderão a saída fornecida em cada exemplo.
No entanto, os exemplos neste tutorial funcionarão com qualquer documento válido de WordprocessingML.
Para criar o documento que este tutorial usa, você precisa ter o Microsoft Office 2007 ou posterior instalado ou
precisa ter o Microsoft Office 2003 com o Microsoft Office Compatibility Pack para formatos de arquivo do
Word, Excel e PowerPoint 2007.
using System;
class Program {
public static void Main(string[] args) {
Console.WriteLine("Hello World");
}
}
Hello World
NOTE
Se você estiver usando o Microsoft Word 2003, selecione Documento do Word 2007 na lista suspensa Salvar
como tipo.
Localizando o estilo de parágrafo padrão (C#)
23/10/2019 • 2 minutes to read • Edit Online
Exemplo
DESCRIÇÃO
O exemplo a seguir abre um documento do Office Open XML WordprocessingML, encontrar as partes do
documento e estilo de pacote em seguida, executa uma consulta que localize o nome padrão de estilo. Para obter
informações sobre pacotes de documento do Office Open XML, bem como as partes que os compõem, consulte
em Detalhes de documentos do Office Open XML WordprocessingML (C#).
A consulta encontrar um nó chamado w:style que tem um atributo chamado w:type com um valor de
“parágrafo”, e também tem um atributo chamado w:default com um valor de “1 ". Haverá porque apenas um nó
XML com esses atributos, a consulta usa o operador de Enumerable.First para converter uma coleção para um
único. Então obtém o valor do atributo com o nome w:styleId .
Este exemplo usa classes do assembly WindowsBase. Ele usa tipos no namespace System.IO.Packaging.
Código
const string fileName = "SampleDoc.docx";
// The following query finds all the paragraphs that have the default style.
string defaultStyle =
(string)(
from style in styleDoc.Root.Elements(w + "style")
where (string)style.Attribute(w + "type") == "paragraph"&&
(string)style.Attribute(w + "default") == "1"
select style
).First().Attribute(w + "styleId");
Comentários
Este exemplo gera a seguinte saída:
Próximas etapas
No exemplo a seguir, você criará uma consulta semelhante que localiza todos os parágrafos em um documento e
seus estilos:
Recuperando os parágrafos e seus estilos (C#)
Recuperando os parágrafos e seus estilos (C#)
23/10/2019 • 6 minutes to read • Edit Online
Nesse exemplo, nós escrevemos uma consulta que recupera os nós de parágrafo de um documento de
WordprocessingML. Também identifica o estilo de cada parágrafo.
Esta consulta é baseada na consulta no exemplo anterior, Localizando o estilo de parágrafo padrão (C#), que
recupera o estilo padrão da lista de estilos. Essa informação é necessária de modo que a consulta pode identificar o
estilo dos parágrafos que não têm um estilo definir explicitamente. Os estilos de parágrafo são definidos por meio
do elemento de w:pPr ; se um parágrafo não contém esse elemento, é formatado com o estilo padrão.
Este tópico explica o significado de algumas partes de consulta, então mostra a consulta como parte de um
exemplo completo, que funciona.
Exemplo
A fonte da consulta para recuperar todos os parágrafos em um documento e seus estilos é a seguinte:
Esta expressão é semelhante à fonte da consulta no exemplo anterior, Localizando o estilo de parágrafo padrão
(C#). A principal diferença é que usa o eixo de Descendants em vez do eixo de Elements . A consulta usa o eixo de
Descendants porque em documentos que têm seções, os parágrafos não serão os filhos diretos do elemento do
corpo; em vez, os parágrafos serão dois níveis para baixo na hierarquia. Usando o eixo de Descendants , o código
funcionará de mesmo se o documento usa seções.
Exemplo
A consulta usa uma cláusula de let para determinar o elemento que contém o nó de estilo. Se não houver
nenhum elemento, então styleNode é definido como null :
A cláusula de let primeiro usa o eixo de Elements para localizar todos os elementos chamados pPr , então usa o
método de extensão de Elements para localizar todos os elementos filho chamados pStyle , e finalmente usa o
operador padrão de consulta de FirstOrDefault para converter a um único. Se a coleção estiver vazia, styleNode é
definido como null . Este é um idioma útil para procurar o nó descendente de pStyle . Observe que se o nó filho
de pPr não existir, o código faz ou falhas lançando uma exceção; em vez disso, styleNode é definido como null ,
que é o comportamento desejado desta cláusula de let .
A consulta em uma coleção de um tipo anônimo com dois membros, StyleName e ParagraphNode .
Exemplo
Este exemplo processa um documento de WordprocessingML, recuperando os nós de parágrafo de um
documento de WordprocessingML. Também identifica o estilo de cada parágrafo. Este exemplo cria nos exemplos
anteriores neste tutorial. A nova consulta é chamada nos comentários no código a seguir.
É possível obter instruções para criar o documento de origem deste exemplo em Criando o documento do Office
Open XML de origem (C#).
Este exemplo usa as classes encontradas no assembly WindowsBase. Ele usa tipos no namespace
System.IO.Packaging.
string defaultStyle =
(string)(
from style in styleDoc.Root.Elements(w + "style")
where (string)style.Attribute(w + "type") == "paragraph"&&
(string)style.Attribute(w + "default") == "1"
select style
).First().Attribute(w + "styleId");
Este exemplo produz a seguinte saída quando aplicado ao documento descrito em Criando o documento do Office
Open XML de origem (C#).
StyleName:Heading1
StyleName:Normal
StyleName:Normal
StyleName:Normal
StyleName:Code
StyleName:Code
StyleName:Code
StyleName:Code
StyleName:Code
StyleName:Code
StyleName:Code
StyleName:Normal
StyleName:Normal
StyleName:Normal
StyleName:Code
Próximas etapas
No próximo tópico, Recuperando o texto dos parágrafos (C#), você criará uma consulta para recuperar o texto dos
parágrafos.
Consulte também
Tutorial: manipulando conteúdo em um documento WordprocessingML (C#)
Recuperando o texto dos parágrafos (C#)
04/11/2019 • 5 minutes to read • Edit Online
Este exemplo é criado com base no exemplo anterior, Recuperando os parágrafos e seus estilos (C#). Esse novo
exemplo recupera o texto de cada parágrafo como uma cadeia de caracteres.
Para recuperar o texto, este exemplo adiciona uma consulta adicional que executa iterações através da coleção de
tipos anônimos e projeta uma nova coleção de um tipo anônimo com a adição de um novo membro, Text . Usa o
operador padrão de consulta de Aggregate para concatenar várias cadeias de caracteres em uma cadeia de
caracteres.
Essa técnica (isto é, principalmente se projetando a uma coleção de um tipo anônimo, usando nessa coleção para
se a projetar uma nova coleção de um tipo anônimo) é uma linguagem comum e útil. Esta consulta pode ter sido
gravados sem projeto para o primeiro tipo anônimo. Entretanto, por causa da avaliação lazy, fazer isso não usa de
poder de processamento adicional. O idioma cria um breves mais objetos na heap, mas isso não prejudica
significativamente o desempenho.
Naturalmente, seria possível escrever uma única consulta que contém a funcionalidade para recuperar os
parágrafos, o estilo de cada parágrafo, e o texto de cada parágrafo. No entanto, geralmente é útil dividir uma
consulta mais complexa em consultas múltiplas porque o código resultante é mais modular e fácil de manter. Além
disso, se você precisar de reutilizar parte de consulta, é mais fácil refatorar se as consultas são gravadas dessa
maneira.
Essas consultas, que são encadeadas juntas, usam o modelo de processamento que é examinado em detalhes no
tópico Tutorial: encadear consultas juntas (C#).
Exemplo
Este exemplo processa um documento de WordprocessingML, determinando o nó do elemento, o nome do estilo,
e o texto de cada parágrafo. Este exemplo cria nos exemplos anteriores neste tutorial. A nova consulta é chamada
nos comentários no código a seguir.
Para obter instruções para criar o documento de origem deste exemplo, consulte Criando o documento do Office
Open XML de origem (C#).
Este exemplo usa classes do assembly WindowsBase. Ele usa tipos no namespace System.IO.Packaging.
string defaultStyle =
(string)(
from style in styleDoc.Root.Elements(w + "style")
where (string)style.Attribute(w + "type") == "paragraph"&&
(string)style.Attribute(w + "default") == "1"
select style
).First().Attribute(w + "styleId");
Próximas etapas
O exemplo a seguir mostra como usar um método de extensão, em vez de Aggregate, para concatenar várias
cadeias de caracteres em uma única cadeia de caracteres.
Refatoração usando um método de extensão (C#)
Consulte também
Tutorial: manipulando conteúdo em um documento WordprocessingML (C#)
Execução adiada e avaliação lenta em LINQ to XML (C#)
Refatoração usando um método de extensão (C#)
23/10/2019 • 5 minutes to read • Edit Online
Este exemplo se baseia no exemplo anterior, Recuperando o texto dos parágrafos (C#), refatorando a concatenação
de cadeias de caracteres usando uma função pura implementada como um método de extensão.
O exemplo anterior usa o operador padrão de consulta de Aggregate para concatenar várias cadeias de caracteres
em uma cadeia de caracteres. No entanto, é mais conveniente escrever um método de extensão para fazer isso,
porque a consulta resultante menor e mais simples.
Exemplo
Este exemplo processa um documento de WordprocessingML, recuperando os parágrafos, o estilo de cada
parágrafo, e o texto de cada parágrafo. Este exemplo cria nos exemplos anteriores neste tutorial.
O exemplo contém várias sobrecargas de método StringConcatenate .
É possível obter instruções para criar o documento de origem deste exemplo em Criando o documento do Office
Open XML de origem (C#).
Este exemplo usa classes do assembly WindowsBase. Ele usa tipos no namespace System.IO.Packaging.
Console.WriteLine("{0}", numbers.StringConcatenate());
Console.WriteLine("{0}", numbers.StringConcatenate(":"));
int[] intNumbers = { 1, 2, 3 };
Console.WriteLine("{0}", intNumbers.StringConcatenate(i => i.ToString()));
Console.WriteLine("{0}", intNumbers.StringConcatenate(i => i.ToString(), ":"));
onetwothree
one:two:three:
123
1:2:3:
Exemplo
Agora, o exemplo pode ser alterado para aproveitar o novo método de extensão:
class Program
{
static void Main(string[] args)
{
const string fileName = "SampleDoc.docx";
string defaultStyle =
(string)(
from style in styleDoc.Root.Elements(w + "style")
where (string)style.Attribute(w + "type") == "paragraph" &&
(string)style.Attribute(w + "default") == "1"
select style
).First().Attribute(w + "styleId");
Este exemplo produz a seguinte saída quando aplicado ao documento descrito em Criando o documento do Office
Open XML de origem (C#).
Observe que essa que refatoração é uma variante de refatoração em uma função pura. O próximo tópico
introduzir a exibição de incluir em funções puras com mais detalhes.
Próximas etapas
O exemplo a seguir mostra como refatorar esse código em outra maneira, usando funções puras:
Refatoração usando uma função pura (Visual Basic)
Consulte também
Tutorial: manipulando conteúdo em um documento WordprocessingML (C#)
Refatoração em funções puras (C#)
Refatoração usando uma função pura (C#)
23/10/2019 • 3 minutes to read • Edit Online
O exemplo a seguir refatora o exemplo anterior, Refatoração usando um método de extensão (C#), para usar uma
função pura. Neste exemplo, o código para localizar o texto de um parágrafo é movido para o método estático puro
ParagraphText .
Exemplo
Este exemplo processa um documento de WordprocessingML, recuperando os nós de parágrafo de um documento
de WordprocessingML. Também identifica o estilo de cada parágrafo. Este exemplo cria nos exemplos anteriores
neste tutorial. O código refactored é chamado nos comentários no código a seguir.
Para obter instruções para criar o documento de origem deste exemplo, consulte Criando o documento do Office
Open XML de origem (C#).
Este exemplo usa classes do assembly WindowsBase. Ele usa tipos no namespace System.IO.Packaging.
class Program
{
// This is a new method that assembles the paragraph text.
public static string ParagraphText(XElement e)
{
{
XNamespace w = e.Name.Namespace;
return e
.Elements(w + "r")
.Elements(w + "t")
.StringConcatenate(element => (string)element);
}
string defaultStyle =
(string)(
from style in styleDoc.Root.Elements(w + "style")
where (string)style.Attribute(w + "type") == "paragraph" &&
(string)style.Attribute(w + "default") == "1"
select style
).First().Attribute(w + "styleId");
Próximas etapas
O exemplo a seguir mostra como projetar XML em uma forma diferente:
Projetando XML em uma forma diferente (C#)
Consulte também
Tutorial: manipulando conteúdo em um documento WordprocessingML (C#)
Refatoração usando um método de extensão (C#)
Refatoração em funções puras (C#)
Projetando XML em uma forma diferente (C#)
23/10/2019 • 4 minutes to read • Edit Online
Este tópico mostra um exemplo de projetar XML que está em uma forma diferente do XML fonte.
Várias transformações típicas XML consistem em consultas encadeadas, como neste exemplo. É comum para
iniciar com alguma forma XML, resultados intermediários de projeto como coleções de tipos anônimos ou tipos
nomeados, e para projetar finalmente nos resultados de novo em XML que está em uma forma totalmente
diferente do XML fonte.
Exemplo
Este exemplo processa um documento de WordprocessingML, recuperando os nós de parágrafo de um
documento de WordprocessingML. Também identifica o estilo e o texto de cada parágrafo. Finalmente, o exemplo
em XML com uma forma diferente. Este exemplo cria nos exemplos anteriores neste tutorial. A nova declaração
que faz a projeção é chamada nos comentários no código a seguir.
Para obter instruções para criar o documento de origem deste exemplo, consulte Criando o documento do Office
Open XML de origem (C#).
Este exemplo usa classes do assembly WindowsBase. Ele usa tipos no namespace System.IO.Packaging.
string defaultStyle =
(string)(
from style in styleDoc.Root.Elements(w + "style")
where (string)style.Attribute(w + "type") == "paragraph" &&
(string)style.Attribute(w + "default") == "1"
select style
).First().Attribute(w + "styleId");
// The following is the new code that projects XML in a new shape.
XElement root = new XElement("Root",
from p in paraWithText
select new XElement("Paragraph",
new XElement("StyleName", p.StyleName),
new XElement("Text", p.Text)
)
);
Console.WriteLine(root);
}
}
Próximas etapas
No exemplo a seguir, você consultará para localizar todo o texto em um documento do Word:
Localizando texto em documentos do Word (C#)
Localizando texto em documentos do Word (C#)
23/10/2019 • 7 minutes to read • Edit Online
Este tópico estende as consultas anteriores para fazer algo útil: localizar todas as ocorrências de uma cadeia de
caracteres no documento.
Exemplo
Este exemplo processa um documento de WordprocessingML para localizar todas as ocorrências de uma parte
específica de texto no documento. Para fazer isso, usamos uma consulta encontrar a cadeia de caracteres “hello
world”. Este exemplo cria nos exemplos anteriores neste tutorial. A nova consulta é chamada nos comentários no
código a seguir.
Para obter instruções para criar o documento de origem deste exemplo, consulte Criando o documento do Office
Open XML de origem (C#).
Este exemplo usa as classes encontradas no assembly WindowsBase. Ele usa tipos no namespace
System.IO.Packaging.
class Program
{
public static string ParagraphText(XElement e)
{
{
XNamespace w = e.Name.Namespace;
return e
.Elements(w + "r")
.Elements(w + "t")
.StringConcatenate(element => (string)element);
}
string defaultStyle =
(string)(
from style in styleDoc.Root.Elements(w + "style")
where (string)style.Attribute(w + "type") == "paragraph" &&
(string)style.Attribute(w + "default") == "1"
select style
).First().Attribute(w + "styleId");
Você pode, é claro, altere a pesquisa para que procura por linhas com um determinado estilo. A consulta a seguir
localiza todas as linhas em branco que têm o estilo de código a seguir:
class Program
{
public static string ParagraphText(XElement e)
{
XNamespace w = e.Name.Namespace;
return e
.Elements(w + "r")
.Elements(w + "t")
.StringConcatenate(element => (string)element);
}
string defaultStyle =
(string)(
(string)(
from style in styleDoc.Root.Elements(w + "style")
where (string)style.Attribute(w + "type") == "paragraph" &&
(string)style.Attribute(w + "default") == "1"
select style
).First().Attribute(w + "styleId");
// Retrieve all paragraphs that have no text and are styled Code.
var blankCodeParagraphs =
from para in paraWithText
where para.Text == "" && para.StyleName == "Code"
select new
{
ParagraphNode = para.ParagraphNode,
StyleName = para.StyleName,
Text = para.Text
};
StyleName:Code ><
Naturalmente, este exemplo pode ser aprimorado de várias maneiras. Por exemplo, nós poderíamos usar
expressões regulares para procurar pelo texto, nós poderíamos iterar através de todos os arquivos do Word em um
diretório específico, e assim por diante.
Observe que este exemplo reproduz funciona bem como se ele foi gravado como uma única consulta. Porque cada
consulta é implementada em um lenta, a forma adiada, cada consulta não produz resultados até que a consulta é
iterada. Para obter mais informações sobre a execução e avaliação lenta, consulte Execução adiada e avaliação lenta
em LINQ to XML (C#).
Próximas etapas
A próxima seção fornece informações sobre documentos de WordprocessingML:
Detalhes de documentos do Office Open XML WordprocessingML (C#)
Consulte também
Tutorial: manipulando conteúdo em um documento WordprocessingML (C#)
Refatoração usando uma função pura (C#)
Execução adiada e avaliação lenta em LINQ to XML (C#)
Documento de WordprocessingML com estilos
23/10/2019 • 4 minutes to read • Edit Online
Um documentos mais complicados de WordprocessingML têm os parágrafos que são formatados com estilos.
Quaisquer notas sobre a composição de documentos de WordprocessingML são úteis. Documentos de
WordprocessingML são armazenados em pacotes. Pacotes têm várias partes (partes tem um significado explícito
quando usadas no contexto de pacotes; essencialmente, as partes são arquivos que são fechados juntos para
entender um pacote). Se um documento contém os parágrafos que são formatados com estilos, haverá uma parte
do documento que contém os parágrafos que possuem os estilos aplicados a eles. Também haverá uma parte de
estilo que contém os estilos que são referenciados pelo documento.
Para acessar pacotes, é importante que você faz isso com relações entre as partes, em vez de usar um caminho
arbitrário. Esse problema está além do escopo do tutorial de Manipulando conteúdo em um documento
WordprocessingML, mas os programas de exemplo que são incluídos neste tutorial demonstram a abordagem
correta.
Este tópico mostra um exemplo de parte de estilo de documento do Office Open XML WordprocessingML.
Exemplo
O exemplo a seguir é XML que compõem a parte de estilo de um documento do Office Open XML
WordprocessingML.
O estilo padrão de parágrafo tem um elemento com a seguir marca de abertura:
Você precisa saber essas informações quando você escreve a consulta para localizar o identificador padrão de
estilo, de modo que a consulta pode identificar o estilo dos parágrafos que têm o estilo padrão.
Observe que esses documentos são muito simples quando comparados para documentos típicos que o Microsoft
Word gerencia. Em muitos casos, o Word salva muitos informações adicionais, formatação extra e metadados.
Além disso, Palavra não formata as linhas para ser facilmente legíveis como neste exemplo; em vez disso, XML é
salvo sem recuo. No entanto, todos os documentos de WordprocessingML compartilham a mesma forma básica
XML. Devido a isso, consultas apresentadas neste tutorial funcionarão com documentos mais complicados.
Este tópico mostra como abrir um documento do Office Open XML e acessar partes dentro deles.
Exemplo
O exemplo a seguir abre um documento do Office Open XML, e imprime a parte do documento e parte de estilo
para o console.
Este exemplo usa classes do assembly WindowsBase. Ele usa tipos no namespace System.IO.Packaging.
const string fileName = "SampleDoc.docx";
Console.WriteLine("TargetUri:{0}", docPackageRelationship.TargetUri);
Console.WriteLine("==================================================================");
Console.WriteLine(xdoc.Root);
Console.WriteLine();
Console.WriteLine("TargetUri:{0}", styleRelation.TargetUri);
Console.WriteLine("==================================================================");
Console.WriteLine(styleDoc.Root);
Console.WriteLine();
}
}
}
Modificação de árvore XML na memória versus
Construção funcional (LINQ to XML) (C#)
23/10/2019 • 6 minutes to read • Edit Online
Modificar uma árvore XML no local é uma abordagem tradicional para alterar a forma de um documento XML.
Um aplicativo típico carregar um documento em um armazenamento de dados como os DOM ou LINQ to XML;
usa uma interface de programação para inserir nós, nós, excluir ou modificar o conteúdo dos nós; e então salva
XML para um arquivo ou fluxo em uma rede.
O LINQ to XML permite outra abordagem que é útil em muitos cenários: construção funcional. Deleites
funcionais de compilação que modificam dados como um problema de transformação, em vez de como
tratamento detalhada de um armazenamento de dados. Se você pode ter uma representação dos dados e a
transformação com eficiência de um formulário para outro, o resultado é o mesmo como se você recebe um
armazenamento de dados e o manipulou de alguma maneira para executar outra forma. Uma chave para a
abordagem de construção funcional é passar os resultados de consultas para os construtores XDocument e
XElement.
Em muitos casos você pode escrever código transformacional em uma fração de tempo que iria para manipular o
armazenamento de dados, e que o código é mais fácil e mais robusto manter. Nesses casos, mesmo que a
abordagem transformacional pode levar mais avançados de processamento, é mais eficiente para modificar
dados. Se um desenvolvedor estiver familiarizado com a abordagem funcional, o código resultante é em muitos
casos mais fácil de entender. É mais fácil de localizar o código que altera cada parte da árvore.
A abordagem onde você altera uma árvore XML no local é mais familiar para muitos desenvolvedores DOM,
enquanto o código escrito usando a abordagem funcional pode parecer estranha a um desenvolvedor que não
entende ainda essa abordagem. Se você precisa apenas fazer uma alteração pequena a uma grande árvore XML,
a abordagem onde você altera uma árvore no lugar em muitos casos terá menos processador central - tempo.
Este tópico fornece um exemplo que é implementado com as duas abordagens.
<Root>
<Child1>Content</Child1>
<Data1>123</Data1>
<Data2>456</Data2>
</Root>
Saída deste exemplo mesmo XML que o primeiro exemplo. No entanto, observe que você pode realmente
consulte a estrutura XML resultante de novo na abordagem funcional. Você pode ver a criação do elemento de
Root , o código que recebe o elemento de Child1 de árvore de origem, e o código que transforma os atributos
de árvore de origem aos elementos na árvore novo.
O exemplo funcional não nesse caso é menor que o primeiro exemplo, e não é realmente mais simples. No
entanto, se você tiver muitas alterações para fazer a uma árvore XML, a abordagem não funcional será muito
complexa e um pouco obtuso. Por outro lado, ao usar a abordagem funcional, você ainda forma apenas XML
desejado, inserindo consultas e expressões apropriadas, para receber dentro do conteúdo desejado. Os passa
funcionais de abordagem código que é mais fácil de manter.
Observe que neste caso a abordagem funcional provavelmente não deseja executar muito bem como a
abordagem de manipulação de árvore. A principal problema é que a abordagem funcional cria um objetos mais
breves. No entanto, as troca são eficiente usar a abordagem funcional permite maior produtividade do
programador.
Este é um exemplo muito simples, mas serve para mostrar a diferença na filosofia entre as duas abordagens. A
abordagem mais funcional fornece ganhos de produtividade para transformar documentos XML maiores.
Adicionando elementos, atributos e nós a uma árvore
XML (C#)
23/10/2019 • 2 minutes to read • Edit Online
Você pode adicionar conteúdo (elementos, atributos, comentários, instruções de processamento, texto e CDATA) a
uma árvore XML existente.
MÉTODO DESCRIÇÃO
Os seguintes métodos adicionam o conteúdo como nós irmãos de um XNode. O nó mais comum ao qual você
adiciona conteúdo irmão é XElement, embora você possa adicionar conteúdo irmão válido a outros tipos de nós
como XText ou XComment.
MÉTODO DESCRIÇÃO
Exemplo
Descrição
O exemplo a seguir cria duas árvores XML e, em seguida, modifica uma das árvores.
Código
XElement srcTree = new XElement("Root",
new XElement("Element1", 1),
new XElement("Element2", 2),
new XElement("Element3", 3),
new XElement("Element4", 4),
new XElement("Element5", 5)
);
XElement xmlTree = new XElement("Root",
new XElement("Child1", 1),
new XElement("Child2", 2),
new XElement("Child3", 3),
new XElement("Child4", 4),
new XElement("Child5", 5)
);
xmlTree.Add(new XElement("NewChild", "new content"));
xmlTree.Add(
from el in srcTree.Elements()
where (int)el > 3
select el
);
// Even though Child9 does not exist in srcTree, the following statement will not
// throw an exception, and nothing will be added to xmlTree.
xmlTree.Add(srcTree.Element("Child9"));
Console.WriteLine(xmlTree);
Comentários
Esse código gera a seguinte saída:
<Root>
<Child1>1</Child1>
<Child2>2</Child2>
<Child3>3</Child3>
<Child4>4</Child4>
<Child5>5</Child5>
<NewChild>new content</NewChild>
<Element4>4</Element4>
<Element5>5</Element5>
</Root>
Modificando elementos, atributos e nós em uma
árvore XML
23/10/2019 • 2 minutes to read • Edit Online
A tabela a seguir resume os métodos e as propriedades que você pode usar para modificar um elemento, seus
elementos filho ou seus atributos.
Os seguintes métodos modificam uma classe XElement.
MÉTODO DESCRIÇÃO
MÉTODO DESCRIÇÃO
MÉTODO DESCRIÇÃO
MÉTODO DESCRIÇÃO
Você pode modificar uma árvore XML, remover elementos, atributos e outros tipos de nós.
Remover um único elemento ou um único atributo de um documento XML é simples. No entanto, ao remover
coleções de elementos ou atributos, você deve primeiro materializar uma coleção em uma lista e depois excluir os
elementos ou os atributos da lista. A melhor abordagem é usar o método de extensão Remove, que fará isso para
você.
O principal motivo para fazer isso é que a maioria das coleções que você recupera de uma árvore XML é gerada
usando a execução adiada. Se você não materializar essas coleções primeiro em uma lista, ou não usar os métodos
de extensão, poderá encontrar uma determinada classe de bugs. Para obter mais informações, consulte Bugs
misturados de código declarativo/código imperativo (LINQ to XML ) (C#).
Os métodos a seguir removem nós e atributos de uma árvore XML.
MÉTODO DESCRIÇÃO
Exemplo
DESCRIÇÃO
Este exemplo demonstra três abordagens para remover elementos. Primeiro, ele remove um único elemento.
Segundo, ele recupera uma coleção de elementos, materializa essa coleção usando o operador Enumerable.ToList e
remove a coleção. Por último, recupera uma coleção de elementos e remove-a usando o método de extensão
Remove.
Para obter mais informações sobre o operador ToList, consulte Convertendo Tipos de Dados (C#).
Código
XElement root = XElement.Parse(@"<Root>
<Child1>
<GrandChild1/>
<GrandChild2/>
<GrandChild3/>
</Child1>
<Child2>
<GrandChild4/>
<GrandChild5/>
<GrandChild6/>
</Child2>
<Child3>
<GrandChild7/>
<GrandChild8/>
<GrandChild9/>
</Child3>
</Root>");
root.Element("Child1").Element("GrandChild1").Remove();
root.Element("Child2").Elements().ToList().Remove();
root.Element("Child3").Elements().Remove();
Console.WriteLine(root);
Comentários
Esse código gera a seguinte saída:
<Root>
<Child1>
<GrandChild2 />
<GrandChild3 />
</Child1>
<Child2 />
<Child3 />
</Root>
Observe que o primeiro elemento neto foi removido de Child1 . Todos os elementos neto foram removidos de
Child2 e de Child3 .
Mantendo pares nome-valor (C#)
23/10/2019 • 4 minutes to read • Edit Online
Muitos aplicativos devem manter informações que são melhor armazenadas como pares de valor/nome. Essas
informações podem ser informações de configuração ou configurações globais. O LINQ to XML contém alguns
métodos que facilitam o trabalho de manter um conjunto de pares de valor/nome. É possível manter informações
como atributos ou como um conjunto de elementos filho.
Uma das diferenças entre manter as informações como atributos ou como elementos filho é que os atributos
possuem a restrição de que pode existir apenas um atributo com um nome específico para um elemento. Essa
limitação não se aplica aos elementos filho.
SetAttributeValue e SetElementValue
Os dois métodos que facilitam o trabalho de manter pares de valor/nome são SetAttributeValue e
SetElementValue. Esses dois métodos possuem semântica semelhante.
SetAttributeValue pode adicionar, modificar ou remover atributos de um elemento.
Ao designar SetAttributeValue com o nome de um atributo inexistente, o método cria um novo atributo e o
adiciona ao elemento especificado.
Ao designar SetAttributeValue com o nome de um atributo existente e algum conteúdo especificado, o
conteúdo do atributo é substituído pelo conteúdo especificado.
Ao designar SetAttributeValue com o nome de um atributo existente e especificar nulo em relação ao
conteúdo, o atributo é removido de seu pai.
SetElementValue pode adicionar, modificar, ou remover elementos filho de um elemento.
Ao designar SetElementValue com o nome de um elemento filho que não existe, o método cria um novo
elemento e o adiciona ao elemento especificado.
Ao designar SetElementValue com o nome de um elemento existente e algum conteúdo especificado, o
conteúdo do elemento é substituído pelo conteúdo especificado.
Ao designar SetElementValue com o nome de um elemento existente e especificar nulo em relação ao
conteúdo, o elemento é removido de seu pai.
Exemplo
O exemplo a seguir cria um elemento sem atributos. Usa o método SetAttributeValue para criar e manter uma lista
de pares de valor/nome.
// Create an element with no content.
XElement root = new XElement("Root");
// Remove DefaultColor.
root.SetAttributeValue("DefaultColor", null);
Console.WriteLine(root);
Exemplo
O exemplo a seguir cria um elemento sem elementos filho. Usa o método SetElementValue para criar e manter
uma lista de pares de valor/nome.
// Remove DefaultColor.
root.SetElementValue("DefaultColor", null);
Console.WriteLine(root);
Consulte também
SetAttributeValue
SetElementValue
Modificando árvores XML (LINQ to XML ) (C#)
Como alterar o namespace para uma árvore XML
inteira (C#)
25/11/2019 • 2 minutes to read • Edit Online
Às vezes você tem que alterar programaticamente ao namespace para um elemento ou atributo. LINQ to XML faz
isso fácil. A propriedade de XElement.Name pode ser definida. A propriedade de XAttribute.Name não pode ser
definida, mas você pode facilmente copiar os atributos em System.Collections.Generic.List<T>, remover os
atributos existentes, e então adiciona novos atributos que estão no novo namespace desejada.
Para obter mais informações, consulte Visão geral de namespaces (LINQ to XML ) (C#).
Exemplo
O código a seguir cria duas árvores XML em qualquer namespace. Altera o namespace de cada uma das árvores, e
as combina em uma única árvore.
XElement tree1 = new XElement("Data",
new XElement("Child", "content",
new XAttribute("MyAttr", "content")
)
);
XElement tree2 = new XElement("Data",
new XElement("Child", "content",
new XAttribute("MyAttr", "content")
)
);
XNamespace aw = "http://www.adventure-works.com";
XNamespace ad = "http://www.adatum.com";
// change the namespace of every element and attribute in the first tree
foreach (XElement el in tree1.DescendantsAndSelf())
{
el.Name = aw.GetName(el.Name.LocalName);
List<XAttribute> atList = el.Attributes().ToList();
el.Attributes().Remove();
foreach (XAttribute at in atList)
el.Add(new XAttribute(aw.GetName(at.Name.LocalName), at.Value));
}
// change the namespace of every element and attribute in the second tree
foreach (XElement el in tree2.DescendantsAndSelf())
{
el.Name = ad.GetName(el.Name.LocalName);
List<XAttribute> atList = el.Attributes().ToList();
el.Attributes().Remove();
foreach (XAttribute at in atList)
el.Add(new XAttribute(ad.GetName(at.Name.LocalName), at.Value));
}
// add attribute namespaces so that the tree will be serialized with
// the aw and ad namespace prefixes
tree1.Add(
new XAttribute(XNamespace.Xmlns + "aw", "http://www.adventure-works.com")
);
tree2.Add(
new XAttribute(XNamespace.Xmlns + "ad", "http://www.adatum.com")
);
// create a new composite tree
XElement root = new XElement("Root",
tree1,
tree2
);
Console.WriteLine(root);
<Root>
<aw:Data xmlns:aw="http://www.adventure-works.com">
<aw:Child aw:MyAttr="content">content</aw:Child>
</aw:Data>
<ad:Data xmlns:ad="http://www.adatum.com">
<ad:Child ad:MyAttr="content">content</ad:Child>
</ad:Data>
</Root>
Desempenho de consultas encadeadas (LINQ to
XML) (C#)
23/10/2019 • 3 minutes to read • Edit Online
Um dos benefícios mais importantes LINQ (e LINQ to XML ) é que as consultas encadeadas podem executar bem
como uma única consulta maior, mais complicada.
Uma consulta encadeada é uma consulta que usa outra consulta como sua fonte. Por exemplo, o seguinte código
simples, query2 tem query1 como sua fonte:
Esta consulta encadeada fornece o mesmo perfil de desempenho que iterando através de uma lista vinculada.
O eixo de Elements tem essencialmente o mesmo desempenho que iterando através de uma lista vinculada.
Elements é implementado como um iterador com execução adiada. Isso significa que faz qualquer trabalho
adicional a iterar através da lista vinculada, como alocar do objeto de iterador e manter controle de estado
de execução. Este trabalho pode ser dividida em duas categorias: o trabalho que é feito no momento no
iterador é configurado, e o trabalho que é feito em cada iteração. O trabalho de configuração é uma pequena
quantidade, fixa de trabalho e o trabalho feito em cada iteração é proporcionalmente para o número de itens
na coleção de origem.
Em query1 , a cláusula de where faz com que a consulta chama o método de Where . Esse método também
é implementado como um iterador. O trabalho de configuração consiste em criar uma instância do
representante que fará referência a expressão lambda, mais a configuração normal para um iterador. A cada
iteração, o representante é chamado para executar o predicado. O trabalho de configuração e trabalho feitos
em cada iteração são semelhantes ao trabalho feito para iterar através do eixo.
Em query1 , a cláusula SELECT faz com que a consulta chama o método de Select . Este método tem o
mesmo perfil de desempenho que o método de Where .
Em query2 , a cláusula de where e a cláusula select têm o mesmo perfil de desempenho que em query1 .
A interação com query2 é portanto diretamente proporcionalmente para o número de itens na fonte da primeira
consulta, ou ser uma das vezes. Um exemplo correspondente do Visual Basic terá o mesmo perfil de desempenho.
Para obter mais informações sobre iteradores, consulte yield.
Para obter um tutorial mais detalhado sobre o encadeamento de consultas, confira Tutorial: Encadeando consultas.
Objetos XName e XNamespace atomizados (LINQ to
XML) (C#)
23/10/2019 • 3 minutes to read • Edit Online
Os objetos XName e XNamespace são atomizados, isto é, se eles contêm o mesmo nome qualificado, referem-se
ao mesmo objeto. Isso resulta em benefícios de desempenho para consultas: Quando você compara dois nomes
atomizados quanto à igualdade, a linguagem intermediária subjacente só precisa determinar se as duas referências
apontam para o mesmo objeto. O código subjacente não tem que fazer as comparações de cadeias de caracteres,
que poderiam demoradas.
Semântica de atomização
A atomização significa que se dois objetos de XName têm o mesmo nome local, e estão no mesmo namespace,
compartilham a mesma instância. Da mesma forma, se dois objetos de XNamespace têm o mesmo URI de
namespace, compartilham a mesma instância.
Para que uma classe permite objetos atomizados, o construtor para a classe deve ser particular, não público. Isso
ocorre porque se foi o construtor público, você pode criar um objeto não atomizado. As classes de XName e de
XNamespace implementam um operador de conversão implícita para converter uma cadeia de caracteres em
XName ou em XNamespace. Isso é como você obtém uma instância desses objetos. Você não pode obter uma
instância usando um construtor, porque o construtor é inacessível.
XName e XNamespace também implementam os operadores de igualdade e desigualdade de, para determinar se
dois objetos que estão sendo comparados são referências para a mesma instância.
Exemplo
O código a seguir cria alguns objetos de XElement e demonstrar-los que os nomes idênticos compartilham a
mesma instância.
if ((object)r1.Name == (object)r2.Name)
Console.WriteLine("r1 and r2 have names that refer to the same instance.");
else
Console.WriteLine("Different");
XName n = "Root";
if ((object)n == (object)r1.Name)
Console.WriteLine("The name of r1 and the name in 'n' refer to the same instance.");
else
Console.WriteLine("Different");
Como mencionado anteriormente, a vantagem de objetos atomizados é que quando você usa um dos métodos do
eixo que recebem XName como um parâmetro, o método do eixo só precisa determinar que dois nomes
referenciam a mesma instância para selecionar os elementos desejados.
O exemplo a seguir XName passa a chamada de método Descendants , que tem em melhor desempenho devido
ao padrão de atomização.
<C1>1</C1>
<C1>1</C1>
Pré-atomização de objetos XName (LINQ to XML)
(C#)
23/10/2019 • 3 minutes to read • Edit Online
Uma maneira para melhorar o desempenho em LINQ to XML é pré-compilação atomizar objetos de XName . a
pré-compilação atomização significa que você atribuir uma cadeia de caracteres a um objeto de XName antes de
criar a árvore XML usando os construtores de classes de XElement e de XAttribute . Em seguida, em vez de passar
uma cadeia de caracteres para o construtor, que usaria a conversão implícita de cadeia de caracteres a XName,
você passa o objeto inicializado de XName .
Isso melhora o desempenho quando você cria uma grande árvore XML em que os nomes específicos são
repetidos. Para fazer isso, você declarar e inicializar objetos de XName antes que você construa a árvore XML e em
seguida, usar objetos de XName em vez de especificar cadeias de caracteres para nomes de elementos e atributos.
Essa técnica pode produzir ganhos significativos de desempenho se você estiver criando um grande número de
elementos ou atributos () com o mesmo nome.
Você deve testar a pré-compilação atomização com seu cenário para decidir se você usar o.
Exemplo
O exemplo a seguir demonstra isso.
Console.WriteLine(root);
<Root>
<Data ID="1">4,100,000</Data>
<Data ID="2">3,700,000</Data>
<Data ID="3">1,150,000</Data>
</Root>
Console.WriteLine(root);
<aw:Root xmlns:aw="http://www.adventure-works.com">
<aw:Data ID="1">4,100,000</aw:Data>
<aw:Data ID="2">3,700,000</aw:Data>
<aw:Data ID="3">1,150,000</aw:Data>
</aw:Root>
O exemplo a seguir é mais semelhante ao que você provavelmente encontrará no mundo real. Nesse exemplo, o
conteúdo do elemento é fornecido por uma consulta:
DateTime t1 = DateTime.Now;
XElement root = new XElement(Root,
from i in System.Linq.Enumerable.Range(1, 100000)
select new XElement(Data,
new XAttribute(ID, i),
i * 5)
);
DateTime t2 = DateTime.Now;
O exemplo anterior executa melhor do que o exemplo, em que os nomes não são atomizados passos:
DateTime t1 = DateTime.Now;
XElement root = new XElement("Root",
from i in System.Linq.Enumerable.Range(1, 100000)
select new XElement("Data",
new XAttribute("ID", i),
i * 5)
);
DateTime t2 = DateTime.Now;
Um de desempenho mais importante beneficia LINQ to XML, diferentemente de XmlDocument, é que as consultas
em LINQ to XML são compiladas estaticamente, enquanto as consultas XPath devem ser interpretado em tempo
de execução. Esse recurso é interna a LINQ to XML, portanto você não precisa executar etapas adicionais para
aproveitá-lo, mas é útil entender a diferença ao escolher entre as duas tecnologias. Este tópico explica a diferença.
XDocument po = XDocument.Load("PurchaseOrders.xml");
IEnumerable<XElement> list1 =
from el in po.Descendants("Address")
where (string)el.Attribute("Type") == "Shipping"
select el;
A expressão de consulta nesse exemplo é reescrita pelo compilador a sintaxe da consulta com base em método. O
exemplo a seguir, que é escrito na sintaxe da consulta com base em método, gerenciar os mesmos resultados que
anterior:
XDocument po = XDocument.Load("PurchaseOrders.xml");
IEnumerable<XElement> list1 =
po
.Descendants("Address")
.Where(el => (string)el.Attribute("Type") == "Shipping");
O método de Where é um método de extensão. Para obter mais informações, consulte Métodos de extensão.
Porque Where é um método de extensão, a consulta anterior é criada como se ele foi gravado como segue:
XDocument po = XDocument.Load("PurchaseOrders.xml");
IEnumerable<XElement> list1 =
System.Linq.Enumerable.Where(
po.Descendants("Address"),
el => (string)el.Attribute("Type") == "Shipping");
NOTE
Esses exemplos são representante de código que o compilador pode escrever. A implementação real pode diferir um pouco
desses exemplos, mas o desempenho será o mesmo ou semelhante a esses exemplos.
Esta consulta retorna as mesmas saída que os exemplos que usam LINQ to XML; a única diferença é que LINQ to
XML recua XML impresso, enquanto XmlDocument não.
No entanto, a abordagem de XmlDocument geralmente não executa bem como LINQ to XML, porque o método
de SelectNodes deve fazer o seguinte internamente todas as vezes nele é chamado:
Analisa a cadeia de caracteres que contém a expressão XPath, quebrando a cadeia de caracteres em tokens.
Valida os tokens para certificar-se de que a expressão XPath é válido.
Converte a expressão em uma árvore de expressão interna.
Itera através de nós, selecionando adequadamente os nós para o conjunto de resultados com base na
avaliação da expressão.
Este é significativamente mais do que o trabalho feito pela consulta correspondente LINQ to XML. A diferença
desempenho específico variam para tipos diferentes de consultas, mas em geral LINQ to XML consultas menos
trabalho e, portanto, executam melhor do que as expressões XPath de avaliação usando XmlDocument.
Anotações LINQ to XML
23/10/2019 • 2 minutes to read • Edit Online
Anotações em LINQ to XML permitem associar qualquer objeto arbitrário de qualquer tipo arbitrário com
qualquer componente XML em uma árvore XML.
Para adicionar uma anotação a um componente XML, como XElement ou XAttribute, você chama o método de
AddAnnotation . Você recupera anotações pelo tipo.
Observe que as anotações não são parte de infoset XML; não são serializados ou não estão desserializados.
Métodos
Você pode usar os seguintes métodos para trabalhar com anotações:
MÉTODO DESCRIÇÃO
Eventos LINQ to XML permitem que você seja notificado quando uma árvore XML é modificada.
Você pode adicionar eventos a uma instância de qualquer XObject. O manipulador de eventos em receberá eventos
para alterações ao XObject e a qualquer um dos seus descendentes. Por exemplo, você pode adicionar um
manipulador de eventos à raiz da árvore, e trata todas as alterações na árvore do manipulador de eventos.
Para exemplos de eventos de LINQ to XML, consulte Changing e Changed.
Tipos e eventos
Você usa os seguintes tipos ao trabalhar com eventos:
TIPO DESCRIÇÃO
Os seguintes eventos são gerados quando você altera uma árvore XML:
EVENTO DESCRIÇÃO
Exemplo
Descrição
Os eventos são úteis quando você deseja manter algumas informações aggregate em uma árvore XML. Por
exemplo, você pode querer mantém um total de fatura que é a soma das linhas de item de fatura. Este exemplo usa
eventos para manter o total de todos os elementos filho no elemento complexo Items .
Código
XElement root = new XElement("Root",
new XElement("Total", "0"),
new XElement("Items")
);
XElement total = root.Element("Total");
XElement items = root.Element("Items");
items.Changed += (object sender, XObjectChangeEventArgs cea) =>
{
switch (cea.ObjectChange)
{
case XObjectChange.Add:
if (sender is XElement)
total.Value = ((int)total + (int)(XElement)sender).ToString();
if (sender is XText)
total.Value = ((int)total + (int)((XText)sender).Parent).ToString();
break;
case XObjectChange.Remove:
if (sender is XElement)
total.Value = ((int)total - (int)(XElement)sender).ToString();
if (sender is XText)
total.Value = ((int)total - Int32.Parse(((XText)sender).Value)).ToString();
break;
}
Console.WriteLine("Changed {0} {1}",
sender.GetType().ToString(), cea.ObjectChange.ToString());
};
items.SetElementValue("Item1", 25);
items.SetElementValue("Item2", 50);
items.SetElementValue("Item2", 75);
items.SetElementValue("Item3", 133);
items.SetElementValue("Item1", null);
items.SetElementValue("Item4", 100);
Console.WriteLine("Total:{0}", (int)total);
Console.WriteLine(root);
Comentários
Esse código gera a seguinte saída:
desenvolvedores deLINQ to XML que precisam geralmente escrever programas como um editor XML, um sistema
uma transformação, ou uma necessidade o gravador de relatório escrever programas que funcionam no nível mais
fino de granularidade dos elementos e atributos. Freqüentemente necessitam de trabalhar no nível de nó, em nós
de manipulação de texto, as instruções de processamento, e os comentários. Este tópico fornece alguns detalhes
sobre programação no nível do nó.
Detalhes do nó
Há um número de detalhes de programação que um programador que funciona a nível do nó deve conhecer.
A propriedade pai de nós de filhos de XDocument é definida como nula
A propriedade de Parent contém XElementpai, não o nó pai. Os nós filho de XDocument não têm nenhum
XElementpai. Seu pai é o documento, assim que a propriedade de Parent para os nós é definida como nula.
O exemplo a seguir demonstra este:
True
True
Console.WriteLine(xmlTree.Nodes().OfType<XText>().Count());
// the following line does not cause the removal of the text node.
textNode.Value = "";
>><<
<Child1></Child1>
<Child2 />
xmlns="http://www.adventure-works.com" IsNamespaceDeclaration:True
xmlns:fc="www.fourthcoffee.com" IsNamespaceDeclaration:True
AnAttribute="abc" IsNamespaceDeclaration:False
<Root/>
<!--a comment-->
", LoadOptions.PreserveWhitespace);
3
0
// this shows that there is only one child node of the document
Console.WriteLine(doc.Nodes().Count());
LINQ to XML contém vários métodos que permitem que você modifique uma árvore XML diretamente. Você
pode adicionar elementos, excluir elementos, modifica o conteúdo de um elemento, adiciona atributos, e assim por
diante. Essa interface de programação é descrita em Modificando árvores XML (LINQ to XML ) (C#) . Se você
estiver iterando com um dos eixos, como Elements, e você está alterando a árvore XML como você itera através do
eixo, você pode acabar com alguns erros estranhas.
Esse problema é às vezes conhecido como “o problema do Dia De Bruxas”.
Definição do problema
Quando você escrever qualquer código usando LINQ que itera através de uma coleção, você estiver escrevendo
código em um estilo declarativo. É mais próximo a descrever o que você deseja, em vez de como deseja que isso
seja feito. Se você escreve o código que 1) obtém o primeiro elemento, 2) testá-la para alguma condição, 3) altera-
a, e 4) coloque-a de novo na lista, então este código seria obrigatório. Você está informando o computador como
deseja que isso seja feito.
Misturar esses estilos de código na mesma operação é o que resulta em problemas. Considere o seguinte:
Suponha que você tenha uma lista vinculada com três itens nele (a, b, e c#):
a -> b -> c
Agora, suponha que você deseja mover através da lista vinculada, adicionando novos itens três (a, b, e c#). Você
deseja a lista vinculada resultante para ter esta aparência:
a -> a' -> b -> b' -> c -> c'
Assim você escreve o código que itera através da lista, e para cada item, adicione um novo item mesmo após ele. O
que acontece são que seu código verá o primeiro elemento de a , e inserção a' após ele. Agora, seu código se o
nó seguir na lista, que agora é a' ! Felizmente adicionar um novo item à lista, a'' .
Como você determinaria este no mundo real? Bem, você pode fazer uma cópia do original associado para listar, e
criar uma lista completamente nova. Ou se você estiver escrevendo código puramente obrigatório, você pode
localizar o primeiro item, adiciona o novo item, e então avanço duas vezes na lista vinculada, adiantando sobre o
elemento que você adicionou.
Adicionar a iterar
Por exemplo, suponha que você deseja escrever qualquer código que para cada elemento em uma árvore, você
deseja criar um elemento duplicado:
<Root>
<A>1</A>
<B>2</B>
<C>3</C>
<A>1</A>
<B>2</B>
<C>3</C>
</Root>
No entanto, isso não faz o que você deseja. Nesta situação, depois que você removesse o primeiro elemento, A, é
removido da árvore XML contida na raiz, e o código no método dos elementos que está fazendo iterar não pode
localizar o elemento seguir.
O código anterior gerencia a saída a seguir:
<Root>
<B>2</B>
<C>3</C>
</Root>
<Root />
Como alternativa, você pode eliminar a iteração completamente chamando RemoveAll no elemento pai:
var z =
from e in root.Elements()
where TestSomeCondition(e)
select DoMyProjection(e);
Esse código de análise precisaria analisar métodos TestSomeCondition e DoMyProjection, e todos os métodos que
esses métodos chamados a partir, para determinar se qualquer código tinha efeitos colaterais. Mas o código de
análise não pode apenas procurar qualquer código que tem efeitos colaterais. Precisaria para selecionar apenas o
código que tinha efeitos colaterais em elementos filho de root nesta situação.
LINQ to XML não tenta fazer uma análise.
Você pode para evitar esses problemas.
Diretrizes
Primeiro, não mistura o código declarativo e obrigatório.
Mesmo se você souber exatamente a semântica das coleções e semântica dos métodos que modificam a árvore
XML, se você escrever qualquer código inteligente que impede essas categorias de problemas, seu código deverá
ser mantido no futuro por outros desenvolvedores, e não podem ser como o espaço livre nos problemas. Se você
mistura estilos declarativo e obrigatórias de codificação, seu código será mais frágil.
Se você escreve o código que materializa uma coleção para que esses problemas são impedidos, observar-la com
comentários apropriadas em seu código, para que os desenvolvedores de aplicativos compreendam o problema.
Segundo, se o desempenho e outras considerações reservam, use somente o código declarativo. Não altere sua
árvore XML existente. Gerencia um novo.
Às vezes você precisará ler arbitrariamente grandes arquivos XML, e escreve seu aplicativo para que os vestígio
de memória do aplicativo seja previsível. Se você tentar preencher uma árvore XML com um grande arquivo
XML, seu uso de memória será proporcionalmente o tamanho do arquivo que é, excessivo. Portanto, você deve
usar uma técnica de streaming em vez disso.
Uma opção é escrever seu aplicativo usando XmlReader. No entanto, talvez você queira usar LINQ para consultar
a árvore XML. Se esse for o caso, você pode escrever seu próprio método personalizado do eixo. Para obter mais
informações, confira Como: Escrever um método de eixo LINQ to XML (C#).
Para escrever seu próprio método do eixo, você escreve um pequeno método que usa XmlReader para ler nós até
que atingiu um dos nós em que você está interessado. O método chama em ReadFrom, que lê de XmlReader e
cria uma instância de um fragmento XML. Resulta em cada fragmento com yield return o método que está
enumerando o método personalizado do eixo. Você pode escrever consultas LINQ no método personalizado do
eixo.
As técnicas de streaming são melhor aplicadas em situações onde você precisa processar o documento de origem
apenas uma vez e você pode processar os elementos na ordem do documento. Determinados operadores de
consulta padrão, como OrderBy, iteram sua origem, coletam todos os dados, classificam e, em seguida, geram
finalmente o primeiro item na sequência. Observe que se você usar um operador de consulta que materialize sua
origem antes de como o primeiro item, você não manterá pegada uma pequena de memória.
Exemplo
Às vezes o problema é apenas um pouco mais interessante. No documento XML a seguir, o consumidor do
método personalizado do eixo também precisa saber o nome do cliente que cada item pertence.
<?xml version="1.0" encoding="utf-8" ?>
<Root>
<Customer>
<Name>A. Datum Corporation</Name>
<Item>
<Key>0001</Key>
</Item>
<Item>
<Key>0002</Key>
</Item>
<Item>
<Key>0003</Key>
</Item>
<Item>
<Key>0004</Key>
</Item>
</Customer>
<Customer>
<Name>Fabrikam, Inc.</Name>
<Item>
<Key>0005</Key>
</Item>
<Item>
<Key>0006</Key>
</Item>
<Item>
<Key>0007</Key>
</Item>
<Item>
<Key>0008</Key>
</Item>
</Customer>
<Customer>
<Name>Southridge Video</Name>
<Item>
<Key>0009</Key>
</Item>
<Item>
<Key>0010</Key>
</Item>
</Customer>
</Root>
A abordagem que este exemplo usa é inspecione também para essas informações de cabeçalho, salvar
informações de cabeçalho, e então cria uma árvore XML pequena que contém informações de cabeçalho e o
detalhe que você está enumerando. O método do eixo resulta nessa nova árvore, pequena XML. A consulta possui
o acesso a informações de cabeçalho bem como para informações detalhadas.
Essa abordagem tem uma pegada pequena de memória. Como cada fragmento XML de detalhe é rendido,
nenhuma referência é mantido para o fragmento anterior, e está disponível para coleta de lixo. Observe que essa
técnica cria uma breve muitos objetos na heap.
O exemplo a seguir mostra como implementar e usar um método personalizado do eixo que passa fragmentos
XML do arquivo especificado por um URI. Esse eixo personalizado é escrito especificamente para que espera um
documento que tem Customer , Name , e os elementos de Item , e que esses elementos serão organizados como
no documento anterior de Source.xml . É uma implementação simplista. Uma implementação mais robusta seria
preparada para analisar um documento válido.
static IEnumerable<XElement> StreamCustomerItem(string uri)
{
using (XmlReader reader = XmlReader.Create(uri))
{
XElement name = null;
XElement item = null;
reader.MoveToContent();
// Parse the file, save header information when encountered, and yield the
// Item XElement objects as they are created.
Às vezes você precisa transformar grandes arquivos XML e escrever seu aplicativo de modo que os requisitos de
memória do aplicativo sejam previsíveis. Se você tentar preencher uma árvore XML com um arquivo XML muito
grande, seu uso de memória será proporcional ao tamanho do arquivo (isto é, excessivo). Portanto, você deve usar
uma técnica de streaming em vez disso.
As técnicas de streaming são melhor aplicadas em situações onde você precisa processar o documento de origem
apenas uma vez e você pode processar os elementos na ordem do documento. Determinados operadores de
consulta padrão, como OrderBy, iteram sua origem, coletam todos os dados, classificam e, em seguida, geram
finalmente o primeiro item na sequência. Observe que se você usar um operador de consulta que materializa sua
origem antes de gerar o primeiro item, você não manterá um requisito pequeno de memória para seu aplicativo.
Mesmo se você usar a técnica descrita em Como: Transmitir fragmentos XML com acesso a informações de
cabeçalho (C#), se você tentar montar uma árvore XML que contenha o documento transformado, o uso de
memória será muito grande.
Há duas principais abordagens. Uma abordagem é usar as características de processamento adiado de
XStreamingElement. Outra abordagem é criar XmlWriter e usar os recursos de LINQ to XML para gravar os
elementos em um XmlWriter. Este tópico demonstra as duas abordagens.
Exemplo
O seguinte exemplo se baseia no exemplo descrito em Como: Transmitir fragmentos XML com acesso a
informações de cabeçalho (C#).
Este exemplo usa os recursos de execução adiada de XStreamingElement para transmitir a saída. Este exemplo
pode transformar um documento muito grande mantendo um requisito pequeno de memória.
Observe que o eixo personalizado ( StreamCustomerItem ) é escrito especificamente para esperar um documento que
tem os elementos Customer , Name e Item , e que esses elementos serão organizados como no documento
Source.xml a seguir. Uma implementação mais robusta, no entanto, seria preparada para analisar um documento
inválido.
Veja a seguir o documento de origem, Source.xml:
<?xml version="1.0" encoding="utf-8" ?>
<Root>
<Customer>
<Name>A. Datum Corporation</Name>
<Item>
<Key>0001</Key>
</Item>
<Item>
<Key>0002</Key>
</Item>
<Item>
<Key>0003</Key>
</Item>
<Item>
<Key>0004</Key>
</Item>
</Customer>
<Customer>
<Name>Fabrikam, Inc.</Name>
<Item>
<Key>0005</Key>
</Item>
<Item>
<Key>0006</Key>
</Item>
<Item>
<Key>0007</Key>
</Item>
<Item>
<Key>0008</Key>
</Item>
</Customer>
<Customer>
<Name>Southridge Video</Name>
<Item>
<Key>0009</Key>
</Item>
<Item>
<Key>0010</Key>
</Item>
</Customer>
</Root>
static IEnumerable<XElement> StreamCustomerItem(string uri)
{
using (XmlReader reader = XmlReader.Create(uri))
{
XElement name = null;
XElement item = null;
reader.MoveToContent();
// Parse the file, save header information when encountered, and yield the
// Item XElement objects as they are created.
Exemplo
O seguinte exemplo também se baseia no exemplo descrito em Como: Transmitir fragmentos XML com acesso a
informações de cabeçalho (C#).
Este exemplo usa o recurso de LINQ to XML para gravar os elementos em um XmlWriter. Este exemplo pode
transformar um documento muito grande mantendo um requisito pequeno de memória.
Observe que o eixo personalizado ( StreamCustomerItem ) é escrito especificamente para esperar um documento que
tem os elementos Customer , Name e Item , e que esses elementos serão organizados como no documento
Source.xml a seguir. Uma implementação mais robusta, no entanto, validaria o documento de origem com um
XSD ou seria preparada para analisar um documento inválido.
Este exemplo usa o mesmo documento de origem, Source.xml, como o exemplo anterior neste tópico. Ele também
produz exatamente a mesma saída.
Usar XStreamingElement para transmitir o XML de saída é preferencial em vez de gravar em um XmlWriter.
static IEnumerable<XElement> StreamCustomerItem(string uri)
{
using (XmlReader reader = XmlReader.Create(uri))
{
XElement name = null;
XElement item = null;
reader.MoveToContent();
// Parse the file, save header information when encountered, and yield the
// Item XElement objects as they are created.
<Root>
<Item>
<Customer>A. Datum Corporation</Customer>
<Key>0001</Key>
</Item>
<Item>
<Customer>A. Datum Corporation</Customer>
<Key>0002</Key>
</Item>
<Item>
<Customer>A. Datum Corporation</Customer>
<Key>0003</Key>
</Item>
<Item>
<Customer>A. Datum Corporation</Customer>
<Key>0004</Key>
</Item>
<Item>
<Customer>Fabrikam, Inc.</Customer>
<Key>0005</Key>
</Item>
<Item>
<Customer>Fabrikam, Inc.</Customer>
<Key>0006</Key>
</Item>
<Item>
<Customer>Fabrikam, Inc.</Customer>
<Key>0007</Key>
</Item>
<Item>
<Customer>Fabrikam, Inc.</Customer>
<Key>0008</Key>
</Item>
<Item>
<Customer>Southridge Video</Customer>
<Key>0009</Key>
</Item>
<Item>
<Customer>Southridge Video</Customer>
<Key>0010</Key>
</Item>
</Root>
Como: Ler e gravar um documento codificado (C#)
23/10/2019 • 2 minutes to read • Edit Online
Para criar um documento XML codificado, adicione um XDeclaration à árvore XML, definindo a codificação para o
nome da página de código desejada.
Qualquer valor retornado por WebName é um valor válido.
Se você ler um documento codificado, a propriedade Encoding será definida para o nome da página de código.
Se você definir Encoding como um nome válido de página de código, o LINQ to XML serializará com a codificação
especificada.
Exemplo
O exemplo a seguir cria dois documentos, um com codificação utf-8 e um com codificação utf-16. Ele em seguida
carrega os documentos e imprime a codificação para o console.
Encoded document:
<?xml version="1.0" encoding="utf-8" standalone="yes"?>
<Root>Content</Root>
Encoded document:
<?xml version="1.0" encoding="utf-16" standalone="yes"?>
<Root>Content</Root>
Consulte também
XDeclaration.Encoding
Usando XSLT para transformar uma árvore XML (C#)
25/11/2019 • 2 minutes to read • Edit Online
Você pode criar uma árvore XML, criar um XmlReader na árvore XML, criar um novo documento e criar um
XmlWriter que gravarão no novo documento. Em seguida, você pode chamar a transformação XSLT, passando
XmlReader e XmlWriter para a transformação. Depois que a transformação for concluída com êxito, a nova árvore
XML será populada com os resultados da transformação.
Exemplo
string xslt = @"<?xml version='1.0'?>
<xsl:stylesheet xmlns:xsl='http://www.w3.org/1999/XSL/Transform' version='1.0'>
<xsl:template match='/Parent'>
<Root>
<C1>
<xsl:value-of select='Child1'/>
</C1>
<C2>
<xsl:value-of select='Child2'/>
</C2>
</Root>
</xsl:template>
</xsl:stylesheet>";
Consulte também
XContainer.CreateWriter
XNode.CreateReader
Como: Usar anotações para transformar árvores
LINQ to XML em um estilo XSLT (C#)
23/10/2019 • 10 minutes to read • Edit Online
As anotações podem ser usadas para facilitar tornam-se de uma árvore XML.
Alguns documentos XML são “centralizado no documento misturado com conteúdo.” Como com documentos,
você não souber necessariamente a forma de nós filho de um elemento. Por exemplo, um nó que contém o texto
pode ter esta aparência:
Para qualquer nó de texto, pode haver qualquer número de <b> filho e elementos de <i> . Essa abordagem
estende-se a diversas outras situações, como páginas que podem conter uma variedade de elementos filho, como
parágrafos normais, parágrafos com marcadores e bitmaps. As células em uma tabela podem conter texto, soltar
para baixo, listas ou bitmaps. Uma das principais características de documento XML é centralizado em que você
não sabe qual elemento filho qualquer elemento específico terá.
Se você deseja transformar elementos em uma árvore onde você não sabe necessariamente muito sobre os filhos
dos elementos que você deseja transformar, então essa abordagem que usa anotações é uma abordagem eficiente.
O resumo de abordagem é:
Primeiro, anotações os elementos na árvore com um elemento de substituição.
Segundo, iterar através da árvore inteira, criando uma nova árvore onde você substitui cada elemento com
a anotação. Este exemplo implementa a iteração e a criação de novo em árvore em uma função chamada
XForm .
XNamespace xf = "http://www.microsoft.com/LinqToXmlTransform/2007";
XName at = xf + "ApplyTransforms";
// The XForm method, shown later in this topic, accomplishes the transform
XElement newRoot = XForm(root);
Console.WriteLine(newRoot);
<Root>
<para>This is a sentence with <b>bold</b> and <i>italic</i> text.</para>
<para>More text.</para>
</Root>
// while adding annotations, you can query the source tree all you want,
// as the tree is not mutated while annotating.
var avg = data.Elements("Data").Select(z => (Decimal)z).Average();
data.AddAnnotation(
new XElement("Root",
new XElement(xf + "ApplyTransforms"),
new XElement("Average", $"{avg:F4}"),
new XElement("Sum",
data
.Elements("Data")
.Select(z => (int)z)
.Sum()
)
)
);
Console.WriteLine("Before Transform");
Console.WriteLine("----------------");
Console.WriteLine(data);
Console.WriteLine();
Console.WriteLine();
// The XForm method, shown later in this topic, accomplishes the transform
XElement newData = XForm(data);
Console.WriteLine("After Transform");
Console.WriteLine("----------------");
Console.WriteLine(newData);
Before Transform
----------------
<Root>
<Data>20</Data>
<Data>10</Data>
<Data>3</Data>
</Root>
After Transform
----------------
<Root>
<Data>20</Data>
<Data>10</Data>
<Data>3</Data>
<Average>11.0000</Average>
<Sum>33</Sum>
</Root>
if (source.Annotation<XElement>() != null)
{
XElement anno = source.Annotation<XElement>();
return new XElement(anno.Name,
anno.Attributes(),
anno
.Nodes()
.Select(
(XNode n) =>
{
XElement annoEl = n as XElement;
if (annoEl != null)
{
if (annoEl.Name == at)
return (object)(
source.Nodes()
.Select(
(XNode n2) =>
{
XElement e2 = n2 as XElement;
if (e2 == null)
return n2;
else
return XForm(e2);
}
)
);
else
return n;
}
else
return n;
}
)
);
}
else
{
return new XElement(source.Name,
source.Attributes(),
source
.Nodes()
.Select(n =>
{
XElement el = n as XElement;
if (el == null)
return n;
else
return XForm(el);
}
)
);
}
}
Exemplo completo
O código a seguir é um exemplo completo que inclui a função de XForm . Inclui alguns usos típicos desse tipo de
transformações:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Xml;
using System.Xml.Linq;
class Program
{
static XNamespace xf = "http://www.microsoft.com/LinqToXmlTransform/2007";
static XName at = xf + "ApplyTransforms";
// replace Other with NewOther, add new child elements around original content
foreach (var el in root.Elements("Other"))
el.AddAnnotation(
new XElement("NewOther",
new XElement("MyNewChild", 1),
// same idea as xsl:apply-templates
new XElement(xf + "ApplyTransforms"),
new XElement("ChildThatComesAfter")
)
);
Console.WriteLine("After Transform");
Console.WriteLine("----------------");
Console.WriteLine(newRoot);
}
}
Before Transform
----------------
<Root Att1="123">
<!--A comment-->
<Child>1</Child>
<Child>2</Child>
<Other>
<GC>3</GC>
<GC>4</GC>
</Other>
<SomeMixedContent>This is <i>an</i> element that <b>has</b> some mixed content</SomeMixedContent>
<AnUnchangedElement>42</AnUnchangedElement>
</Root>
After Transform
----------------
<Root Att1="123">
<!--A comment-->
<NewChild>1</NewChild>
<NewChild>2</NewChild>
<NewOther>
<MyNewChild>1</MyNewChild>
<GrandChild ANewAttribute="999">3</GrandChild>
<GC>4</GC>
<ChildThatComesAfter />
</NewOther>
<MixedContent>This is <Italic>an</Italic> element that <Bold>has</Bold> some mixed content</MixedContent>
<AnUnchangedElement>42</AnUnchangedElement>
</Root>
Como: Serializar usando XmlSerializer (C#)
23/10/2019 • 2 minutes to read • Edit Online
Exemplo
O exemplo a seguir cria um número de objetos que contêm objetos de XElement . Serializar-los a um fluxo de
memória, e desserializa os de fluxo de memória.
using System;
using System.IO;
using System.Linq;
using System.Xml;
using System.Xml.Serialization;
using System.Xml.Linq;
public XElementContainer()
{
member = XLinqTest.CreateXElement();
}
public XElementNullContainer()
{
}
}
class XLinqTest
{
static void Main(string[] args)
{
Test<XElementNullContainer>(new XElementNullContainer());
Test<XElement>(CreateXElement());
Test<XElementContainer>(new XElementContainer());
}
Exemplo
O exemplo a seguir cria um número de objetos que contêm objetos de XElement . Serializar-los em arquivos de
texto, e desserializa nos arquivos de texto.
using System;
using System.Xml;
using System.Xml.Linq;
using System.IO;
using System.Runtime.Serialization;
[DataContract]
public class XElementContainer
{
[DataMember]
public XElement member;
public XElementContainer()
{
member = XLinqTest.CreateXElement();
}
}
[DataContract]
public class XElementNullContainer
{
[DataMember]
public XElement member;
public XElementNullContainer()
{
member = null;
}
}
Este tópico descreve problemas de segurança associadas LINQ to XML. Além disso, fornece alguma orientação
para a exposição de segurança de abrandamento.
O arquivo XML a seguir é usado em vários exemplos na documentação do LINQ to XML. Este arquivo é uma
ordem de compra típica.
PurchaseOrder.xml
<?xml version="1.0"?>
<PurchaseOrder PurchaseOrderNumber="99503" OrderDate="1999-10-20">
<Address Type="Shipping">
<Name>Ellen Adams</Name>
<Street>123 Maple Street</Street>
<City>Mill Valley</City>
<State>CA</State>
<Zip>10999</Zip>
<Country>USA</Country>
</Address>
<Address Type="Billing">
<Name>Tai Yee</Name>
<Street>8 Oak Avenue</Street>
<City>Old Town</City>
<State>PA</State>
<Zip>95819</Zip>
<Country>USA</Country>
</Address>
<DeliveryNotes>Please leave packages in shed by driveway.</DeliveryNotes>
<Items>
<Item PartNumber="872-AA">
<ProductName>Lawnmower</ProductName>
<Quantity>1</Quantity>
<USPrice>148.95</USPrice>
<Comment>Confirm this is electric</Comment>
</Item>
<Item PartNumber="926-AA">
<ProductName>Baby Monitor</ProductName>
<Quantity>2</Quantity>
<USPrice>39.98</USPrice>
<ShipDate>1999-05-21</ShipDate>
</Item>
</Items>
</PurchaseOrder>
Arquivo XML de exemplo: Ordem de compra típica
em um namespace
23/10/2019 • 2 minutes to read • Edit Online
O arquivo XML a seguir é usado em vários exemplos na documentação do LINQ to XML. Este arquivo é uma
ordem de compra típica. XML é em um namespace.
PurchaseOrderInNamespace.xml
<?xml version="1.0"?>
<aw:PurchaseOrder
aw:PurchaseOrderNumber="99503"
aw:OrderDate="1999-10-20"
xmlns:aw="http://www.adventure-works.com">
<aw:Address aw:Type="Shipping">
<aw:Name>Ellen Adams</aw:Name>
<aw:Street>123 Maple Street</aw:Street>
<aw:City>Mill Valley</aw:City>
<aw:State>CA</aw:State>
<aw:Zip>10999</aw:Zip>
<aw:Country>USA</aw:Country>
</aw:Address>
<aw:Address aw:Type="Billing">
<aw:Name>Tai Yee</aw:Name>
<aw:Street>8 Oak Avenue</aw:Street>
<aw:City>Old Town</aw:City>
<aw:State>PA</aw:State>
<aw:Zip>95819</aw:Zip>
<aw:Country>USA</aw:Country>
</aw:Address>
<aw:DeliveryNotes>Please leave packages in shed by driveway.</aw:DeliveryNotes>
<aw:Items>
<aw:Item aw:PartNumber="872-AA">
<aw:ProductName>Lawnmower</aw:ProductName>
<aw:Quantity>1</aw:Quantity>
<aw:USPrice>148.95</aw:USPrice>
<aw:Comment>Confirm this is electric</aw:Comment>
</aw:Item>
<aw:Item aw:PartNumber="926-AA">
<aw:ProductName>Baby Monitor</aw:ProductName>
<aw:Quantity>2</aw:Quantity>
<aw:USPrice>39.98</aw:USPrice>
<aw:ShipDate>1999-05-21</aw:ShipDate>
</aw:Item>
</aw:Items>
</aw:PurchaseOrder>
Arquivo XML de exemplo: Várias ordens de compra
(LINQ to XML)
23/10/2019 • 2 minutes to read • Edit Online
O arquivo XML a seguir é usado em vários exemplos na documentação do LINQ to XML. Este arquivo contém
várias ordens de compra.
PurchaseOrders.xml
<?xml version="1.0"?>
<PurchaseOrders>
<PurchaseOrder PurchaseOrderNumber="99503" OrderDate="1999-10-20">
<Address Type="Shipping">
<Name>Ellen Adams</Name>
<Street>123 Maple Street</Street>
<City>Mill Valley</City>
<State>CA</State>
<Zip>10999</Zip>
<Country>USA</Country>
</Address>
<Address Type="Billing">
<Name>Tai Yee</Name>
<Street>8 Oak Avenue</Street>
<City>Old Town</City>
<State>PA</State>
<Zip>95819</Zip>
<Country>USA</Country>
</Address>
<DeliveryNotes>Please leave packages in shed by driveway.</DeliveryNotes>
<Items>
<Item PartNumber="872-AA">
<ProductName>Lawnmower</ProductName>
<Quantity>1</Quantity>
<USPrice>148.95</USPrice>
<Comment>Confirm this is electric</Comment>
</Item>
<Item PartNumber="926-AA">
<ProductName>Baby Monitor</ProductName>
<Quantity>2</Quantity>
<USPrice>39.98</USPrice>
<ShipDate>1999-05-21</ShipDate>
</Item>
</Items>
</PurchaseOrder>
<PurchaseOrder PurchaseOrderNumber="99505" OrderDate="1999-10-22">
<Address Type="Shipping">
<Name>Cristian Osorio</Name>
<Street>456 Main Street</Street>
<City>Buffalo</City>
<State>NY</State>
<Zip>98112</Zip>
<Country>USA</Country>
</Address>
<Address Type="Billing">
<Name>Cristian Osorio</Name>
<Street>456 Main Street</Street>
<City>Buffalo</City>
<State>NY</State>
<Zip>98112</Zip>
<Country>USA</Country>
<Country>USA</Country>
</Address>
<DeliveryNotes>Please notify me before shipping.</DeliveryNotes>
<Items>
<Item PartNumber="456-NM">
<ProductName>Power Supply</ProductName>
<Quantity>1</Quantity>
<USPrice>45.99</USPrice>
</Item>
</Items>
</PurchaseOrder>
<PurchaseOrder PurchaseOrderNumber="99504" OrderDate="1999-10-22">
<Address Type="Shipping">
<Name>Jessica Arnold</Name>
<Street>4055 Madison Ave</Street>
<City>Seattle</City>
<State>WA</State>
<Zip>98112</Zip>
<Country>USA</Country>
</Address>
<Address Type="Billing">
<Name>Jessica Arnold</Name>
<Street>4055 Madison Ave</Street>
<City>Buffalo</City>
<State>NY</State>
<Zip>98112</Zip>
<Country>USA</Country>
</Address>
<Items>
<Item PartNumber="898-AZ">
<ProductName>Computer Keyboard</ProductName>
<Quantity>1</Quantity>
<USPrice>29.99</USPrice>
</Item>
<Item PartNumber="898-AM">
<ProductName>Wireless Mouse</ProductName>
<Quantity>1</Quantity>
<USPrice>14.99</USPrice>
</Item>
</Items>
</PurchaseOrder>
</PurchaseOrders>
Arquivo XML de exemplo: Várias ordens de compra
em um namespace
23/10/2019 • 2 minutes to read • Edit Online
O arquivo XML a seguir é usado em vários exemplos na documentação do LINQ to XML. Este arquivo contém
várias ordens de compra. XML é em um namespace.
PurchaseOrdersInNamespace.xml
<?xml version="1.0" encoding="utf-8"?>
<aw:PurchaseOrders xmlns:aw="http://www.adventure-works.com">
<aw:PurchaseOrder aw:PurchaseOrderNumber="99503" aw:OrderDate="1999-10-20">
<aw:Address aw:Type="Shipping">
<aw:Name>Ellen Adams</aw:Name>
<aw:Street>123 Maple Street</aw:Street>
<aw:City>Mill Valley</aw:City>
<aw:State>CA</aw:State>
<aw:Zip>10999</aw:Zip>
<aw:Country>USA</aw:Country>
</aw:Address>
<aw:Address aw:Type="Billing">
<aw:Name>Tai Yee</aw:Name>
<aw:Street>8 Oak Avenue</aw:Street>
<aw:City>Old Town</aw:City>
<aw:State>PA</aw:State>
<aw:Zip>95819</aw:Zip>
<aw:Country>USA</aw:Country>
</aw:Address>
<aw:DeliveryNotes>Please leave packages in shed by driveway.</aw:DeliveryNotes>
<aw:Items>
<aw:Item aw:PartNumber="872-AA">
<aw:ProductName>Lawnmower</aw:ProductName>
<aw:Quantity>1</aw:Quantity>
<aw:USPrice>148.95</aw:USPrice>
<aw:Comment>Confirm this is electric</aw:Comment>
</aw:Item>
<aw:Item aw:PartNumber="926-AA">
<aw:ProductName>Baby Monitor</aw:ProductName>
<aw:Quantity>2</aw:Quantity>
<aw:USPrice>39.98</aw:USPrice>
<aw:ShipDate>1999-05-21</aw:ShipDate>
</aw:Item>
</aw:Items>
</aw:PurchaseOrder>
<aw:PurchaseOrder aw:PurchaseOrderNumber="99505" aw:OrderDate="1999-10-22">
<aw:Address aw:Type="Shipping">
<aw:Name>Cristian Osorio</aw:Name>
<aw:Street>456 Main Street</aw:Street>
<aw:City>Buffalo</aw:City>
<aw:State>NY</aw:State>
<aw:Zip>98112</aw:Zip>
<aw:Country>USA</aw:Country>
</aw:Address>
<aw:Address aw:Type="Billing">
<aw:Name>Cristian Osorio</aw:Name>
<aw:Street>456 Main Street</aw:Street>
<aw:City>Buffalo</aw:City>
<aw:State>NY</aw:State>
<aw:Zip>98112</aw:Zip>
<aw:Country>USA</aw:Country>
<aw:Country>USA</aw:Country>
</aw:Address>
<aw:DeliveryNotes>Please notify me before shipping.</aw:DeliveryNotes>
<aw:Items>
<aw:Item aw:PartNumber="456-NM">
<aw:ProductName>Power Supply</aw:ProductName>
<aw:Quantity>1</aw:Quantity>
<aw:USPrice>45.99</aw:USPrice>
</aw:Item>
</aw:Items>
</aw:PurchaseOrder>
<aw:PurchaseOrder aw:PurchaseOrderNumber="99504" aw:OrderDate="1999-10-22">
<aw:Address aw:Type="Shipping">
<aw:Name>Jessica Arnold</aw:Name>
<aw:Street>4055 Madison Ave</aw:Street>
<aw:City>Seattle</aw:City>
<aw:State>WA</aw:State>
<aw:Zip>98112</aw:Zip>
<aw:Country>USA</aw:Country>
</aw:Address>
<aw:Address aw:Type="Billing">
<aw:Name>Jessica Arnold</aw:Name>
<aw:Street>4055 Madison Ave</aw:Street>
<aw:City>Buffalo</aw:City>
<aw:State>NY</aw:State>
<aw:Zip>98112</aw:Zip>
<aw:Country>USA</aw:Country>
</aw:Address>
<aw:Items>
<aw:Item aw:PartNumber="898-AZ">
<aw:ProductName>Computer Keyboard</aw:ProductName>
<aw:Quantity>1</aw:Quantity>
<aw:USPrice>29.99</aw:USPrice>
</aw:Item>
<aw:Item aw:PartNumber="898-AM">
<aw:ProductName>Wireless Mouse</aw:ProductName>
<aw:Quantity>1</aw:Quantity>
<aw:USPrice>14.99</aw:USPrice>
</aw:Item>
</aw:Items>
</aw:PurchaseOrder>
</aw:PurchaseOrders>
Arquivo XML de exemplo: Configuração de teste
(LINQ to XML)
23/10/2019 • 2 minutes to read • Edit Online
O arquivo XML a seguir é usado em vários exemplos na documentação do LINQ to XML. Este é um arquivo de
configuração de teste.
TestConfig.xml
<?xml version="1.0"?>
<Tests>
<Test TestId="0001" TestType="CMD">
<Name>Convert number to string</Name>
<CommandLine>Examp1.EXE</CommandLine>
<Input>1</Input>
<Output>One</Output>
</Test>
<Test TestId="0002" TestType="CMD">
<Name>Find succeeding characters</Name>
<CommandLine>Examp2.EXE</CommandLine>
<Input>abc</Input>
<Output>def</Output>
</Test>
<Test TestId="0003" TestType="GUI">
<Name>Convert multiple numbers to strings</Name>
<CommandLine>Examp2.EXE /Verbose</CommandLine>
<Input>123</Input>
<Output>One Two Three</Output>
</Test>
<Test TestId="0004" TestType="GUI">
<Name>Find correlated key</Name>
<CommandLine>Examp3.EXE</CommandLine>
<Input>a1</Input>
<Output>b1</Output>
</Test>
<Test TestId="0005" TestType="GUI">
<Name>Count characters</Name>
<CommandLine>FinalExamp.EXE</CommandLine>
<Input>This is a test</Input>
<Output>14</Output>
</Test>
<Test TestId="0006" TestType="GUI">
<Name>Another Test</Name>
<CommandLine>Examp2.EXE</CommandLine>
<Input>Test Input</Input>
<Output>10</Output>
</Test>
</Tests>
Arquivo XML de exemplo: Configuração de teste em
um namespace
23/10/2019 • 2 minutes to read • Edit Online
O arquivo XML a seguir é usado em vários exemplos na documentação do LINQ to XML. Este é um arquivo de
configuração de teste. XML é em um namespace.
TestConfigInNamespace.xml
<?xml version="1.0"?>
<Tests xmlns="http://www.adatum.com">
<Test TestId="0001" TestType="CMD">
<Name>Convert number to string</Name>
<CommandLine>Examp1.EXE</CommandLine>
<Input>1</Input>
<Output>One</Output>
</Test>
<Test TestId="0002" TestType="CMD">
<Name>Find succeeding characters</Name>
<CommandLine>Examp2.EXE</CommandLine>
<Input>abc</Input>
<Output>def</Output>
</Test>
<Test TestId="0003" TestType="GUI">
<Name>Convert multiple numbers to strings</Name>
<CommandLine>Examp2.EXE /Verbose</CommandLine>
<Input>123</Input>
<Output>One Two Three</Output>
</Test>
<Test TestId="0004" TestType="GUI">
<Name>Find correlated key</Name>
<CommandLine>Examp3.EXE</CommandLine>
<Input>a1</Input>
<Output>b1</Output>
</Test>
<Test TestId="0005" TestType="GUI">
<Name>Count characters</Name>
<CommandLine>FinalExamp.EXE</CommandLine>
<Input>This is a test</Input>
<Output>14</Output>
</Test>
<Test TestId="0006" TestType="GUI">
<Name>Another Test</Name>
<CommandLine>Examp2.EXE</CommandLine>
<Input>Test Input</Input>
<Output>10</Output>
</Test>
</Tests>
Arquivo XML de exemplo: Clientes e ordens (LINQ
to XML)
23/10/2019 • 4 minutes to read • Edit Online
O arquivo XML a seguir é usado em vários exemplos na documentação do LINQ to XML. Este arquivo
contém clientes e pedidos.
O tópico Arquivo XSD de exemplo: Clientes e ordens contém um XSD que pode ser usado para validar esse
documento. Usa os recursos de XSD xs:key e xs:keyref para estabelecer que o atributo CustomerID do
elemento Customer é uma chave, e para estabelecer uma relação entre o elemento CustomerID em cada
elemento Order , e o atributo CustomerID em cada elemento Customer .
Para obter um exemplo de como escrever consultas LINQ que aproveitam essa relação usando a cláusula
Join , confira Como: Unir duas coleções ( LINQ to XML ) ( C#).
CustomersOrders.xml
<?xml version="1.0" encoding="utf-8"?>
<Root>
<Customers>
<Customer CustomerID="GREAL">
<CompanyName>Great Lakes Food Market</CompanyName>
<ContactName>Howard Snyder</ContactName>
<ContactTitle>Marketing Manager</ContactTitle>
<Phone>(503) 555-7555</Phone>
<FullAddress>
<Address>2732 Baker Blvd.</Address>
<City>Eugene</City>
<Region>OR</Region>
<PostalCode>97403</PostalCode>
<Country>USA</Country>
</FullAddress>
</Customer>
<Customer CustomerID="HUNGC">
<CompanyName>Hungry Coyote Import Store</CompanyName>
<ContactName>Yoshi Latimer</ContactName>
<ContactTitle>Sales Representative</ContactTitle>
<Phone>(503) 555-6874</Phone>
<Fax>(503) 555-2376</Fax>
<FullAddress>
<Address>City Center Plaza 516 Main St.</Address>
<City>Elgin</City>
<Region>OR</Region>
<PostalCode>97827</PostalCode>
<Country>USA</Country>
</FullAddress>
</Customer>
<Customer CustomerID="LAZYK">
<CompanyName>Lazy K Kountry Store</CompanyName>
<ContactName>John Steel</ContactName>
<ContactTitle>Marketing Manager</ContactTitle>
<Phone>(509) 555-7969</Phone>
<Fax>(509) 555-6221</Fax>
<FullAddress>
<Address>12 Orchestra Terrace</Address>
<City>Walla Walla</City>
<Region>WA</Region>
<PostalCode>99362</PostalCode>
<Country>USA</Country>
</FullAddress>
</Customer>
<Customer CustomerID="LETSS">
<CompanyName>Let's Stop N Shop</CompanyName>
<ContactName>Jaime Yorres</ContactName>
<ContactTitle>Owner</ContactTitle>
<Phone>(415) 555-5938</Phone>
<FullAddress>
<Address>87 Polk St. Suite 5</Address>
<City>San Francisco</City>
<Region>CA</Region>
<PostalCode>94117</PostalCode>
<Country>USA</Country>
</FullAddress>
</Customer>
</Customers>
<Orders>
<Order>
<CustomerID>GREAL</CustomerID>
<EmployeeID>6</EmployeeID>
<OrderDate>1997-05-06T00:00:00</OrderDate>
<RequiredDate>1997-05-20T00:00:00</RequiredDate>
<ShipInfo ShippedDate="1997-05-09T00:00:00">
<ShipVia>2</ShipVia>
<Freight>3.35</Freight>
<ShipName>Great Lakes Food Market</ShipName>
<ShipAddress>2732 Baker Blvd.</ShipAddress>
<ShipCity>Eugene</ShipCity>
<ShipRegion>OR</ShipRegion>
<ShipPostalCode>97403</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
<Order>
<CustomerID>GREAL</CustomerID>
<EmployeeID>8</EmployeeID>
<OrderDate>1997-07-04T00:00:00</OrderDate>
<RequiredDate>1997-08-01T00:00:00</RequiredDate>
<ShipInfo ShippedDate="1997-07-14T00:00:00">
<ShipVia>2</ShipVia>
<Freight>4.42</Freight>
<ShipName>Great Lakes Food Market</ShipName>
<ShipAddress>2732 Baker Blvd.</ShipAddress>
<ShipCity>Eugene</ShipCity>
<ShipRegion>OR</ShipRegion>
<ShipPostalCode>97403</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
<Order>
<CustomerID>GREAL</CustomerID>
<EmployeeID>1</EmployeeID>
<OrderDate>1997-07-31T00:00:00</OrderDate>
<RequiredDate>1997-08-28T00:00:00</RequiredDate>
<ShipInfo ShippedDate="1997-08-05T00:00:00">
<ShipVia>2</ShipVia>
<Freight>116.53</Freight>
<ShipName>Great Lakes Food Market</ShipName>
<ShipAddress>2732 Baker Blvd.</ShipAddress>
<ShipCity>Eugene</ShipCity>
<ShipRegion>OR</ShipRegion>
<ShipPostalCode>97403</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
<Order>
<CustomerID>GREAL</CustomerID>
<EmployeeID>4</EmployeeID>
<EmployeeID>4</EmployeeID>
<OrderDate>1997-07-31T00:00:00</OrderDate>
<RequiredDate>1997-08-28T00:00:00</RequiredDate>
<ShipInfo ShippedDate="1997-08-04T00:00:00">
<ShipVia>2</ShipVia>
<Freight>18.53</Freight>
<ShipName>Great Lakes Food Market</ShipName>
<ShipAddress>2732 Baker Blvd.</ShipAddress>
<ShipCity>Eugene</ShipCity>
<ShipRegion>OR</ShipRegion>
<ShipPostalCode>97403</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
<Order>
<CustomerID>GREAL</CustomerID>
<EmployeeID>6</EmployeeID>
<OrderDate>1997-09-04T00:00:00</OrderDate>
<RequiredDate>1997-10-02T00:00:00</RequiredDate>
<ShipInfo ShippedDate="1997-09-10T00:00:00">
<ShipVia>1</ShipVia>
<Freight>57.15</Freight>
<ShipName>Great Lakes Food Market</ShipName>
<ShipAddress>2732 Baker Blvd.</ShipAddress>
<ShipCity>Eugene</ShipCity>
<ShipRegion>OR</ShipRegion>
<ShipPostalCode>97403</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
<Order>
<CustomerID>GREAL</CustomerID>
<EmployeeID>3</EmployeeID>
<OrderDate>1997-09-25T00:00:00</OrderDate>
<RequiredDate>1997-10-23T00:00:00</RequiredDate>
<ShipInfo ShippedDate="1997-09-30T00:00:00">
<ShipVia>3</ShipVia>
<Freight>76.13</Freight>
<ShipName>Great Lakes Food Market</ShipName>
<ShipAddress>2732 Baker Blvd.</ShipAddress>
<ShipCity>Eugene</ShipCity>
<ShipRegion>OR</ShipRegion>
<ShipPostalCode>97403</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
<Order>
<CustomerID>GREAL</CustomerID>
<EmployeeID>4</EmployeeID>
<OrderDate>1998-01-06T00:00:00</OrderDate>
<RequiredDate>1998-02-03T00:00:00</RequiredDate>
<ShipInfo ShippedDate="1998-02-04T00:00:00">
<ShipVia>2</ShipVia>
<Freight>719.78</Freight>
<ShipName>Great Lakes Food Market</ShipName>
<ShipAddress>2732 Baker Blvd.</ShipAddress>
<ShipCity>Eugene</ShipCity>
<ShipRegion>OR</ShipRegion>
<ShipPostalCode>97403</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
<Order>
<CustomerID>GREAL</CustomerID>
<EmployeeID>3</EmployeeID>
<OrderDate>1998-03-09T00:00:00</OrderDate>
<RequiredDate>1998-04-06T00:00:00</RequiredDate>
<ShipInfo ShippedDate="1998-03-18T00:00:00">
<ShipVia>2</ShipVia>
<Freight>33.68</Freight>
<Freight>33.68</Freight>
<ShipName>Great Lakes Food Market</ShipName>
<ShipAddress>2732 Baker Blvd.</ShipAddress>
<ShipCity>Eugene</ShipCity>
<ShipRegion>OR</ShipRegion>
<ShipPostalCode>97403</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
<Order>
<CustomerID>GREAL</CustomerID>
<EmployeeID>3</EmployeeID>
<OrderDate>1998-04-07T00:00:00</OrderDate>
<RequiredDate>1998-05-05T00:00:00</RequiredDate>
<ShipInfo ShippedDate="1998-04-15T00:00:00">
<ShipVia>2</ShipVia>
<Freight>25.19</Freight>
<ShipName>Great Lakes Food Market</ShipName>
<ShipAddress>2732 Baker Blvd.</ShipAddress>
<ShipCity>Eugene</ShipCity>
<ShipRegion>OR</ShipRegion>
<ShipPostalCode>97403</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
<Order>
<CustomerID>GREAL</CustomerID>
<EmployeeID>4</EmployeeID>
<OrderDate>1998-04-22T00:00:00</OrderDate>
<RequiredDate>1998-05-20T00:00:00</RequiredDate>
<ShipInfo>
<ShipVia>3</ShipVia>
<Freight>18.84</Freight>
<ShipName>Great Lakes Food Market</ShipName>
<ShipAddress>2732 Baker Blvd.</ShipAddress>
<ShipCity>Eugene</ShipCity>
<ShipRegion>OR</ShipRegion>
<ShipPostalCode>97403</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
<Order>
<CustomerID>GREAL</CustomerID>
<EmployeeID>4</EmployeeID>
<OrderDate>1998-04-30T00:00:00</OrderDate>
<RequiredDate>1998-06-11T00:00:00</RequiredDate>
<ShipInfo>
<ShipVia>3</ShipVia>
<Freight>14.01</Freight>
<ShipName>Great Lakes Food Market</ShipName>
<ShipAddress>2732 Baker Blvd.</ShipAddress>
<ShipCity>Eugene</ShipCity>
<ShipRegion>OR</ShipRegion>
<ShipPostalCode>97403</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
<Order>
<CustomerID>HUNGC</CustomerID>
<EmployeeID>3</EmployeeID>
<OrderDate>1996-12-06T00:00:00</OrderDate>
<RequiredDate>1997-01-03T00:00:00</RequiredDate>
<ShipInfo ShippedDate="1996-12-09T00:00:00">
<ShipVia>2</ShipVia>
<Freight>20.12</Freight>
<ShipName>Hungry Coyote Import Store</ShipName>
<ShipAddress>City Center Plaza 516 Main St.</ShipAddress>
<ShipCity>Elgin</ShipCity>
<ShipRegion>OR</ShipRegion>
<ShipPostalCode>97827</ShipPostalCode>
<ShipPostalCode>97827</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
<Order>
<CustomerID>HUNGC</CustomerID>
<EmployeeID>1</EmployeeID>
<OrderDate>1996-12-25T00:00:00</OrderDate>
<RequiredDate>1997-01-22T00:00:00</RequiredDate>
<ShipInfo ShippedDate="1997-01-03T00:00:00">
<ShipVia>3</ShipVia>
<Freight>30.34</Freight>
<ShipName>Hungry Coyote Import Store</ShipName>
<ShipAddress>City Center Plaza 516 Main St.</ShipAddress>
<ShipCity>Elgin</ShipCity>
<ShipRegion>OR</ShipRegion>
<ShipPostalCode>97827</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
<Order>
<CustomerID>HUNGC</CustomerID>
<EmployeeID>3</EmployeeID>
<OrderDate>1997-01-15T00:00:00</OrderDate>
<RequiredDate>1997-02-12T00:00:00</RequiredDate>
<ShipInfo ShippedDate="1997-01-24T00:00:00">
<ShipVia>1</ShipVia>
<Freight>0.2</Freight>
<ShipName>Hungry Coyote Import Store</ShipName>
<ShipAddress>City Center Plaza 516 Main St.</ShipAddress>
<ShipCity>Elgin</ShipCity>
<ShipRegion>OR</ShipRegion>
<ShipPostalCode>97827</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
<Order>
<CustomerID>HUNGC</CustomerID>
<EmployeeID>4</EmployeeID>
<OrderDate>1997-07-16T00:00:00</OrderDate>
<RequiredDate>1997-08-13T00:00:00</RequiredDate>
<ShipInfo ShippedDate="1997-07-21T00:00:00">
<ShipVia>1</ShipVia>
<Freight>45.13</Freight>
<ShipName>Hungry Coyote Import Store</ShipName>
<ShipAddress>City Center Plaza 516 Main St.</ShipAddress>
<ShipCity>Elgin</ShipCity>
<ShipRegion>OR</ShipRegion>
<ShipPostalCode>97827</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
<Order>
<CustomerID>HUNGC</CustomerID>
<EmployeeID>8</EmployeeID>
<OrderDate>1997-09-08T00:00:00</OrderDate>
<RequiredDate>1997-10-06T00:00:00</RequiredDate>
<ShipInfo ShippedDate="1997-10-15T00:00:00">
<ShipVia>1</ShipVia>
<Freight>111.29</Freight>
<ShipName>Hungry Coyote Import Store</ShipName>
<ShipAddress>City Center Plaza 516 Main St.</ShipAddress>
<ShipCity>Elgin</ShipCity>
<ShipRegion>OR</ShipRegion>
<ShipPostalCode>97827</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
<Order>
<CustomerID>LAZYK</CustomerID>
<CustomerID>LAZYK</CustomerID>
<EmployeeID>1</EmployeeID>
<OrderDate>1997-03-21T00:00:00</OrderDate>
<RequiredDate>1997-04-18T00:00:00</RequiredDate>
<ShipInfo ShippedDate="1997-04-10T00:00:00">
<ShipVia>3</ShipVia>
<Freight>7.48</Freight>
<ShipName>Lazy K Kountry Store</ShipName>
<ShipAddress>12 Orchestra Terrace</ShipAddress>
<ShipCity>Walla Walla</ShipCity>
<ShipRegion>WA</ShipRegion>
<ShipPostalCode>99362</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
<Order>
<CustomerID>LAZYK</CustomerID>
<EmployeeID>8</EmployeeID>
<OrderDate>1997-05-22T00:00:00</OrderDate>
<RequiredDate>1997-06-19T00:00:00</RequiredDate>
<ShipInfo ShippedDate="1997-06-26T00:00:00">
<ShipVia>2</ShipVia>
<Freight>11.92</Freight>
<ShipName>Lazy K Kountry Store</ShipName>
<ShipAddress>12 Orchestra Terrace</ShipAddress>
<ShipCity>Walla Walla</ShipCity>
<ShipRegion>WA</ShipRegion>
<ShipPostalCode>99362</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
<Order>
<CustomerID>LETSS</CustomerID>
<EmployeeID>1</EmployeeID>
<OrderDate>1997-06-25T00:00:00</OrderDate>
<RequiredDate>1997-07-23T00:00:00</RequiredDate>
<ShipInfo ShippedDate="1997-07-04T00:00:00">
<ShipVia>2</ShipVia>
<Freight>13.73</Freight>
<ShipName>Let's Stop N Shop</ShipName>
<ShipAddress>87 Polk St. Suite 5</ShipAddress>
<ShipCity>San Francisco</ShipCity>
<ShipRegion>CA</ShipRegion>
<ShipPostalCode>94117</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
<Order>
<CustomerID>LETSS</CustomerID>
<EmployeeID>8</EmployeeID>
<OrderDate>1997-10-27T00:00:00</OrderDate>
<RequiredDate>1997-11-24T00:00:00</RequiredDate>
<ShipInfo ShippedDate="1997-11-05T00:00:00">
<ShipVia>2</ShipVia>
<Freight>51.44</Freight>
<ShipName>Let's Stop N Shop</ShipName>
<ShipAddress>87 Polk St. Suite 5</ShipAddress>
<ShipCity>San Francisco</ShipCity>
<ShipRegion>CA</ShipRegion>
<ShipPostalCode>94117</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
<Order>
<CustomerID>LETSS</CustomerID>
<EmployeeID>6</EmployeeID>
<OrderDate>1997-11-10T00:00:00</OrderDate>
<RequiredDate>1997-12-08T00:00:00</RequiredDate>
<ShipInfo ShippedDate="1997-11-21T00:00:00">
<ShipVia>2</ShipVia>
<ShipVia>2</ShipVia>
<Freight>45.97</Freight>
<ShipName>Let's Stop N Shop</ShipName>
<ShipAddress>87 Polk St. Suite 5</ShipAddress>
<ShipCity>San Francisco</ShipCity>
<ShipRegion>CA</ShipRegion>
<ShipPostalCode>94117</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
<Order>
<CustomerID>LETSS</CustomerID>
<EmployeeID>4</EmployeeID>
<OrderDate>1998-02-12T00:00:00</OrderDate>
<RequiredDate>1998-03-12T00:00:00</RequiredDate>
<ShipInfo ShippedDate="1998-02-13T00:00:00">
<ShipVia>2</ShipVia>
<Freight>90.97</Freight>
<ShipName>Let's Stop N Shop</ShipName>
<ShipAddress>87 Polk St. Suite 5</ShipAddress>
<ShipCity>San Francisco</ShipCity>
<ShipRegion>CA</ShipRegion>
<ShipPostalCode>94117</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
</Orders>
</Root>
Arquivo de exemplo XSD: Clientes e ordens
23/10/2019 • 2 minutes to read • Edit Online
O arquivo XSD a seguir é usado em vários exemplos na documentação do LINQ to XML. Esse arquivo contém
uma definição de esquema para Arquivo XML de exemplo: Clientes e ordens (LINQ to XML ). O esquema usa os
recursos xs:key e xs:keyref de XSD para estabelecer que o atributo CustomerID do elemento Customer é
uma chave e para estabelecer uma relação entre o elemento CustomerID em cada elemento Order e o atributo
CustomerID em cada elemento Customer .
Para obter um exemplo de como escrever consultas LINQ que aproveitam essa relação usando a cláusula Join ,
confira Como: Unir duas coleções (LINQ to XML ) (C#).
CustomersOrders.xsd
<?xml version="1.0" encoding="utf-8" ?>
<xs:schema xmlns:xs="http://www.w3.org/2001/XMLSchema">
<xs:element name='Root'>
<xs:complexType>
<xs:sequence>
<xs:element name='Customers'>
<xs:complexType>
<xs:sequence>
<xs:element name='Customer' type='CustomerType' minOccurs='0' maxOccurs='unbounded' />
</xs:sequence>
</xs:complexType>
</xs:element>
<xs:element name='Orders'>
<xs:complexType>
<xs:sequence>
<xs:element name='Order' type='OrderType' minOccurs='0' maxOccurs='unbounded' />
</xs:sequence>
</xs:complexType>
</xs:element>
</xs:sequence>
</xs:complexType>
<xs:key name='CustomerIDKey'>
<xs:selector xpath='Customers/Customer'/>
<xs:field xpath='@CustomerID'/>
</xs:key>
<xs:keyref name='CustomerIDKeyRef' refer='CustomerIDKey'>
<xs:selector xpath='Orders/Order'/>
<xs:field xpath='CustomerID'/>
</xs:keyref>
</xs:element>
<xs:complexType name='CustomerType'>
<xs:sequence>
<xs:element name='CompanyName' type='xs:string'/>
<xs:element name='ContactName' type='xs:string'/>
<xs:element name='ContactTitle' type='xs:string'/>
<xs:element name='Phone' type='xs:string'/>
<xs:element name='Fax' minOccurs='0' type='xs:string'/>
<xs:element name='FullAddress' type='AddressType'/>
</xs:sequence>
<xs:attribute name='CustomerID' type='xs:token'/>
</xs:complexType>
<xs:complexType name='AddressType'>
<xs:sequence>
<xs:element name='Address' type='xs:string'/>
<xs:element name='City' type='xs:string'/>
<xs:element name='Region' type='xs:string'/>
<xs:element name='Region' type='xs:string'/>
<xs:element name='PostalCode' type='xs:string' />
<xs:element name='Country' type='xs:string'/>
</xs:sequence>
<xs:attribute name='CustomerID' type='xs:token'/>
</xs:complexType>
<xs:complexType name='OrderType'>
<xs:sequence>
<xs:element name='CustomerID' type='xs:token'/>
<xs:element name='EmployeeID' type='xs:token'/>
<xs:element name='OrderDate' type='xs:dateTime'/>
<xs:element name='RequiredDate' type='xs:dateTime'/>
<xs:element name='ShipInfo' type='ShipInfoType'/>
</xs:sequence>
</xs:complexType>
<xs:complexType name='ShipInfoType'>
<xs:sequence>
<xs:element name='ShipVia' type='xs:integer'/>
<xs:element name='Freight' type='xs:decimal'/>
<xs:element name='ShipName' type='xs:string'/>
<xs:element name='ShipAddress' type='xs:string'/>
<xs:element name='ShipCity' type='xs:string'/>
<xs:element name='ShipRegion' type='xs:string'/>
<xs:element name='ShipPostalCode' type='xs:string'/>
<xs:element name='ShipCountry' type='xs:string'/>
</xs:sequence>
<xs:attribute name='ShippedDate' type='xs:dateTime'/>
</xs:complexType>
</xs:schema>
Arquivo XML de exemplo: Clientes e ordens em um
namespace
23/10/2019 • 3 minutes to read • Edit Online
O arquivo XML a seguir é usado em vários exemplos na documentação do LINQ to XML. Este arquivo contém
clientes e pedidos. XML é em um namespace.
CustomersOrdersInNamespace.xml
<?xml version="1.0" encoding="utf-8"?>
<Root xmlns="http://www.adventure-works.com">
<Customers>
<Customer CustomerID="GREAL">
<CompanyName>Great Lakes Food Market</CompanyName>
<ContactName>Howard Snyder</ContactName>
<ContactTitle>Marketing Manager</ContactTitle>
<Phone>(503) 555-7555</Phone>
<FullAddress>
<Address>2732 Baker Blvd.</Address>
<City>Eugene</City>
<Region>OR</Region>
<PostalCode>97403</PostalCode>
<Country>USA</Country>
</FullAddress>
</Customer>
<Customer CustomerID="HUNGC">
<CompanyName>Hungry Coyote Import Store</CompanyName>
<ContactName>Yoshi Latimer</ContactName>
<ContactTitle>Sales Representative</ContactTitle>
<Phone>(503) 555-6874</Phone>
<Fax>(503) 555-2376</Fax>
<FullAddress>
<Address>City Center Plaza 516 Main St.</Address>
<City>Elgin</City>
<Region>OR</Region>
<PostalCode>97827</PostalCode>
<Country>USA</Country>
</FullAddress>
</Customer>
<Customer CustomerID="LAZYK">
<CompanyName>Lazy K Kountry Store</CompanyName>
<ContactName>John Steel</ContactName>
<ContactTitle>Marketing Manager</ContactTitle>
<Phone>(509) 555-7969</Phone>
<Fax>(509) 555-6221</Fax>
<FullAddress>
<Address>12 Orchestra Terrace</Address>
<City>Walla Walla</City>
<Region>WA</Region>
<PostalCode>99362</PostalCode>
<Country>USA</Country>
</FullAddress>
</Customer>
<Customer CustomerID="LETSS">
<CompanyName>Let's Stop N Shop</CompanyName>
<ContactName>Jaime Yorres</ContactName>
<ContactTitle>Owner</ContactTitle>
<Phone>(415) 555-5938</Phone>
<FullAddress>
<Address>87 Polk St. Suite 5</Address>
<Address>87 Polk St. Suite 5</Address>
<City>San Francisco</City>
<Region>CA</Region>
<PostalCode>94117</PostalCode>
<Country>USA</Country>
</FullAddress>
</Customer>
</Customers>
<Orders>
<Order>
<CustomerID>GREAL</CustomerID>
<EmployeeID>6</EmployeeID>
<OrderDate>1997-05-06T00:00:00</OrderDate>
<RequiredDate>1997-05-20T00:00:00</RequiredDate>
<ShipInfo ShippedDate="1997-05-09T00:00:00">
<ShipVia>2</ShipVia>
<Freight>3.35</Freight>
<ShipName>Great Lakes Food Market</ShipName>
<ShipAddress>2732 Baker Blvd.</ShipAddress>
<ShipCity>Eugene</ShipCity>
<ShipRegion>OR</ShipRegion>
<ShipPostalCode>97403</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
<Order>
<CustomerID>GREAL</CustomerID>
<EmployeeID>8</EmployeeID>
<OrderDate>1997-07-04T00:00:00</OrderDate>
<RequiredDate>1997-08-01T00:00:00</RequiredDate>
<ShipInfo ShippedDate="1997-07-14T00:00:00">
<ShipVia>2</ShipVia>
<Freight>4.42</Freight>
<ShipName>Great Lakes Food Market</ShipName>
<ShipAddress>2732 Baker Blvd.</ShipAddress>
<ShipCity>Eugene</ShipCity>
<ShipRegion>OR</ShipRegion>
<ShipPostalCode>97403</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
<Order>
<CustomerID>GREAL</CustomerID>
<EmployeeID>1</EmployeeID>
<OrderDate>1997-07-31T00:00:00</OrderDate>
<RequiredDate>1997-08-28T00:00:00</RequiredDate>
<ShipInfo ShippedDate="1997-08-05T00:00:00">
<ShipVia>2</ShipVia>
<Freight>116.53</Freight>
<ShipName>Great Lakes Food Market</ShipName>
<ShipAddress>2732 Baker Blvd.</ShipAddress>
<ShipCity>Eugene</ShipCity>
<ShipRegion>OR</ShipRegion>
<ShipPostalCode>97403</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
<Order>
<CustomerID>GREAL</CustomerID>
<EmployeeID>4</EmployeeID>
<OrderDate>1997-07-31T00:00:00</OrderDate>
<RequiredDate>1997-08-28T00:00:00</RequiredDate>
<ShipInfo ShippedDate="1997-08-04T00:00:00">
<ShipVia>2</ShipVia>
<Freight>18.53</Freight>
<ShipName>Great Lakes Food Market</ShipName>
<ShipAddress>2732 Baker Blvd.</ShipAddress>
<ShipCity>Eugene</ShipCity>
<ShipRegion>OR</ShipRegion>
<ShipPostalCode>97403</ShipPostalCode>
<ShipPostalCode>97403</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
<Order>
<CustomerID>GREAL</CustomerID>
<EmployeeID>6</EmployeeID>
<OrderDate>1997-09-04T00:00:00</OrderDate>
<RequiredDate>1997-10-02T00:00:00</RequiredDate>
<ShipInfo ShippedDate="1997-09-10T00:00:00">
<ShipVia>1</ShipVia>
<Freight>57.15</Freight>
<ShipName>Great Lakes Food Market</ShipName>
<ShipAddress>2732 Baker Blvd.</ShipAddress>
<ShipCity>Eugene</ShipCity>
<ShipRegion>OR</ShipRegion>
<ShipPostalCode>97403</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
<Order>
<CustomerID>GREAL</CustomerID>
<EmployeeID>3</EmployeeID>
<OrderDate>1997-09-25T00:00:00</OrderDate>
<RequiredDate>1997-10-23T00:00:00</RequiredDate>
<ShipInfo ShippedDate="1997-09-30T00:00:00">
<ShipVia>3</ShipVia>
<Freight>76.13</Freight>
<ShipName>Great Lakes Food Market</ShipName>
<ShipAddress>2732 Baker Blvd.</ShipAddress>
<ShipCity>Eugene</ShipCity>
<ShipRegion>OR</ShipRegion>
<ShipPostalCode>97403</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
<Order>
<CustomerID>GREAL</CustomerID>
<EmployeeID>4</EmployeeID>
<OrderDate>1998-01-06T00:00:00</OrderDate>
<RequiredDate>1998-02-03T00:00:00</RequiredDate>
<ShipInfo ShippedDate="1998-02-04T00:00:00">
<ShipVia>2</ShipVia>
<Freight>719.78</Freight>
<ShipName>Great Lakes Food Market</ShipName>
<ShipAddress>2732 Baker Blvd.</ShipAddress>
<ShipCity>Eugene</ShipCity>
<ShipRegion>OR</ShipRegion>
<ShipPostalCode>97403</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
<Order>
<CustomerID>GREAL</CustomerID>
<EmployeeID>3</EmployeeID>
<OrderDate>1998-03-09T00:00:00</OrderDate>
<RequiredDate>1998-04-06T00:00:00</RequiredDate>
<ShipInfo ShippedDate="1998-03-18T00:00:00">
<ShipVia>2</ShipVia>
<Freight>33.68</Freight>
<ShipName>Great Lakes Food Market</ShipName>
<ShipAddress>2732 Baker Blvd.</ShipAddress>
<ShipCity>Eugene</ShipCity>
<ShipRegion>OR</ShipRegion>
<ShipPostalCode>97403</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
<Order>
<CustomerID>GREAL</CustomerID>
<CustomerID>GREAL</CustomerID>
<EmployeeID>3</EmployeeID>
<OrderDate>1998-04-07T00:00:00</OrderDate>
<RequiredDate>1998-05-05T00:00:00</RequiredDate>
<ShipInfo ShippedDate="1998-04-15T00:00:00">
<ShipVia>2</ShipVia>
<Freight>25.19</Freight>
<ShipName>Great Lakes Food Market</ShipName>
<ShipAddress>2732 Baker Blvd.</ShipAddress>
<ShipCity>Eugene</ShipCity>
<ShipRegion>OR</ShipRegion>
<ShipPostalCode>97403</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
<Order>
<CustomerID>GREAL</CustomerID>
<EmployeeID>4</EmployeeID>
<OrderDate>1998-04-22T00:00:00</OrderDate>
<RequiredDate>1998-05-20T00:00:00</RequiredDate>
<ShipInfo>
<ShipVia>3</ShipVia>
<Freight>18.84</Freight>
<ShipName>Great Lakes Food Market</ShipName>
<ShipAddress>2732 Baker Blvd.</ShipAddress>
<ShipCity>Eugene</ShipCity>
<ShipRegion>OR</ShipRegion>
<ShipPostalCode>97403</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
<Order>
<CustomerID>GREAL</CustomerID>
<EmployeeID>4</EmployeeID>
<OrderDate>1998-04-30T00:00:00</OrderDate>
<RequiredDate>1998-06-11T00:00:00</RequiredDate>
<ShipInfo>
<ShipVia>3</ShipVia>
<Freight>14.01</Freight>
<ShipName>Great Lakes Food Market</ShipName>
<ShipAddress>2732 Baker Blvd.</ShipAddress>
<ShipCity>Eugene</ShipCity>
<ShipRegion>OR</ShipRegion>
<ShipPostalCode>97403</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
<Order>
<CustomerID>HUNGC</CustomerID>
<EmployeeID>3</EmployeeID>
<OrderDate>1996-12-06T00:00:00</OrderDate>
<RequiredDate>1997-01-03T00:00:00</RequiredDate>
<ShipInfo ShippedDate="1996-12-09T00:00:00">
<ShipVia>2</ShipVia>
<Freight>20.12</Freight>
<ShipName>Hungry Coyote Import Store</ShipName>
<ShipAddress>City Center Plaza 516 Main St.</ShipAddress>
<ShipCity>Elgin</ShipCity>
<ShipRegion>OR</ShipRegion>
<ShipPostalCode>97827</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
<Order>
<CustomerID>HUNGC</CustomerID>
<EmployeeID>1</EmployeeID>
<OrderDate>1996-12-25T00:00:00</OrderDate>
<RequiredDate>1997-01-22T00:00:00</RequiredDate>
<ShipInfo ShippedDate="1997-01-03T00:00:00">
<ShipVia>3</ShipVia>
<Freight>30.34</Freight>
<ShipName>Hungry Coyote Import Store</ShipName>
<ShipAddress>City Center Plaza 516 Main St.</ShipAddress>
<ShipCity>Elgin</ShipCity>
<ShipRegion>OR</ShipRegion>
<ShipPostalCode>97827</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
<Order>
<CustomerID>HUNGC</CustomerID>
<EmployeeID>3</EmployeeID>
<OrderDate>1997-01-15T00:00:00</OrderDate>
<RequiredDate>1997-02-12T00:00:00</RequiredDate>
<ShipInfo ShippedDate="1997-01-24T00:00:00">
<ShipVia>1</ShipVia>
<Freight>0.2</Freight>
<ShipName>Hungry Coyote Import Store</ShipName>
<ShipAddress>City Center Plaza 516 Main St.</ShipAddress>
<ShipCity>Elgin</ShipCity>
<ShipRegion>OR</ShipRegion>
<ShipPostalCode>97827</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
<Order>
<CustomerID>HUNGC</CustomerID>
<EmployeeID>4</EmployeeID>
<OrderDate>1997-07-16T00:00:00</OrderDate>
<RequiredDate>1997-08-13T00:00:00</RequiredDate>
<ShipInfo ShippedDate="1997-07-21T00:00:00">
<ShipVia>1</ShipVia>
<Freight>45.13</Freight>
<ShipName>Hungry Coyote Import Store</ShipName>
<ShipAddress>City Center Plaza 516 Main St.</ShipAddress>
<ShipCity>Elgin</ShipCity>
<ShipRegion>OR</ShipRegion>
<ShipPostalCode>97827</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
<Order>
<CustomerID>HUNGC</CustomerID>
<EmployeeID>8</EmployeeID>
<OrderDate>1997-09-08T00:00:00</OrderDate>
<RequiredDate>1997-10-06T00:00:00</RequiredDate>
<ShipInfo ShippedDate="1997-10-15T00:00:00">
<ShipVia>1</ShipVia>
<Freight>111.29</Freight>
<ShipName>Hungry Coyote Import Store</ShipName>
<ShipAddress>City Center Plaza 516 Main St.</ShipAddress>
<ShipCity>Elgin</ShipCity>
<ShipRegion>OR</ShipRegion>
<ShipPostalCode>97827</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
<Order>
<CustomerID>LAZYK</CustomerID>
<EmployeeID>1</EmployeeID>
<OrderDate>1997-03-21T00:00:00</OrderDate>
<RequiredDate>1997-04-18T00:00:00</RequiredDate>
<ShipInfo ShippedDate="1997-04-10T00:00:00">
<ShipVia>3</ShipVia>
<Freight>7.48</Freight>
<ShipName>Lazy K Kountry Store</ShipName>
<ShipAddress>12 Orchestra Terrace</ShipAddress>
<ShipCity>Walla Walla</ShipCity>
<ShipRegion>WA</ShipRegion>
<ShipPostalCode>99362</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
<Order>
<CustomerID>LAZYK</CustomerID>
<EmployeeID>8</EmployeeID>
<OrderDate>1997-05-22T00:00:00</OrderDate>
<RequiredDate>1997-06-19T00:00:00</RequiredDate>
<ShipInfo ShippedDate="1997-06-26T00:00:00">
<ShipVia>2</ShipVia>
<Freight>11.92</Freight>
<ShipName>Lazy K Kountry Store</ShipName>
<ShipAddress>12 Orchestra Terrace</ShipAddress>
<ShipCity>Walla Walla</ShipCity>
<ShipRegion>WA</ShipRegion>
<ShipPostalCode>99362</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
<Order>
<CustomerID>LETSS</CustomerID>
<EmployeeID>1</EmployeeID>
<OrderDate>1997-06-25T00:00:00</OrderDate>
<RequiredDate>1997-07-23T00:00:00</RequiredDate>
<ShipInfo ShippedDate="1997-07-04T00:00:00">
<ShipVia>2</ShipVia>
<Freight>13.73</Freight>
<ShipName>Let's Stop N Shop</ShipName>
<ShipAddress>87 Polk St. Suite 5</ShipAddress>
<ShipCity>San Francisco</ShipCity>
<ShipRegion>CA</ShipRegion>
<ShipPostalCode>94117</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
<Order>
<CustomerID>LETSS</CustomerID>
<EmployeeID>8</EmployeeID>
<OrderDate>1997-10-27T00:00:00</OrderDate>
<RequiredDate>1997-11-24T00:00:00</RequiredDate>
<ShipInfo ShippedDate="1997-11-05T00:00:00">
<ShipVia>2</ShipVia>
<Freight>51.44</Freight>
<ShipName>Let's Stop N Shop</ShipName>
<ShipAddress>87 Polk St. Suite 5</ShipAddress>
<ShipCity>San Francisco</ShipCity>
<ShipRegion>CA</ShipRegion>
<ShipPostalCode>94117</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
<Order>
<CustomerID>LETSS</CustomerID>
<EmployeeID>6</EmployeeID>
<OrderDate>1997-11-10T00:00:00</OrderDate>
<RequiredDate>1997-12-08T00:00:00</RequiredDate>
<ShipInfo ShippedDate="1997-11-21T00:00:00">
<ShipVia>2</ShipVia>
<Freight>45.97</Freight>
<ShipName>Let's Stop N Shop</ShipName>
<ShipAddress>87 Polk St. Suite 5</ShipAddress>
<ShipCity>San Francisco</ShipCity>
<ShipRegion>CA</ShipRegion>
<ShipPostalCode>94117</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
</Order>
<Order>
<CustomerID>LETSS</CustomerID>
<EmployeeID>4</EmployeeID>
<OrderDate>1998-02-12T00:00:00</OrderDate>
<RequiredDate>1998-03-12T00:00:00</RequiredDate>
<ShipInfo ShippedDate="1998-02-13T00:00:00">
<ShipVia>2</ShipVia>
<Freight>90.97</Freight>
<ShipName>Let's Stop N Shop</ShipName>
<ShipAddress>87 Polk St. Suite 5</ShipAddress>
<ShipCity>San Francisco</ShipCity>
<ShipRegion>CA</ShipRegion>
<ShipPostalCode>94117</ShipPostalCode>
<ShipCountry>USA</ShipCountry>
</ShipInfo>
</Order>
</Orders>
</Root>
Arquivo XML de exemplo: Dados numéricos (LINQ
to XML)
23/10/2019 • 2 minutes to read • Edit Online
O arquivo XML a seguir é usado em vários exemplos na documentação do LINQ to XML. Este arquivo contém
dados numéricos para somar, especifique intermediária, e agrupamento.
Data.xml
<Root>
<TaxRate>7.25</TaxRate>
<Data>
<Category>A</Category>
<Quantity>3</Quantity>
<Price>24.50</Price>
</Data>
<Data>
<Category>B</Category>
<Quantity>1</Quantity>
<Price>89.99</Price>
</Data>
<Data>
<Category>A</Category>
<Quantity>5</Quantity>
<Price>4.95</Price>
</Data>
<Data>
<Category>A</Category>
<Quantity>3</Quantity>
<Price>66.00</Price>
</Data>
<Data>
<Category>B</Category>
<Quantity>10</Quantity>
<Price>.99</Price>
</Data>
<Data>
<Category>A</Category>
<Quantity>15</Quantity>
<Price>29.00</Price>
</Data>
<Data>
<Category>B</Category>
<Quantity>8</Quantity>
<Price>6.99</Price>
</Data>
</Root>
Arquivo XML de exemplo: Dados numéricos em um
namespace
23/10/2019 • 2 minutes to read • Edit Online
O arquivo XML a seguir é usado em vários exemplos na documentação do LINQ to XML. Este arquivo contém
dados numéricos para somar, especifique intermediária, e agrupamento. XML é em um namespace.
Dados
<Root xmlns='http://www.adatum.com'>
<TaxRate>7.25</TaxRate>
<Data>
<Category>A</Category>
<Quantity>3</Quantity>
<Price>24.50</Price>
</Data>
<Data>
<Category>B</Category>
<Quantity>1</Quantity>
<Price>89.99</Price>
</Data>
<Data>
<Category>A</Category>
<Quantity>5</Quantity>
<Price>4.95</Price>
</Data>
<Data>
<Category>A</Category>
<Quantity>3</Quantity>
<Price>66.00</Price>
</Data>
<Data>
<Category>B</Category>
<Quantity>10</Quantity>
<Price>.99</Price>
</Data>
<Data>
<Category>A</Category>
<Quantity>15</Quantity>
<Price>29.00</Price>
</Data>
<Data>
<Category>B</Category>
<Quantity>8</Quantity>
<Price>6.99</Price>
</Data>
</Root>
Arquivo XML de exemplo: Livros (LINQ to XML)
23/10/2019 • 2 minutes to read • Edit Online
O arquivo XML a seguir é usado em vários exemplos na documentação do LINQ to XML. O arquivo contém
informações sobre livros.
books.xml
<?xml version="1.0"?>
<Catalog>
<Book id="bk101">
<Author>Garghentini, Davide</Author>
<Title>XML Developer's Guide</Title>
<Genre>Computer</Genre>
<Price>44.95</Price>
<PublishDate>2000-10-01</PublishDate>
<Description>An in-depth look at creating applications
with XML.</Description>
</Book>
<Book id="bk102">
<Author>Garcia, Debra</Author>
<Title>Midnight Rain</Title>
<Genre>Fantasy</Genre>
<Price>5.95</Price>
<PublishDate>2000-12-16</PublishDate>
<Description>A former architect battles corporate zombies,
an evil sorceress, and her own childhood to become queen
of the world.</Description>
</Book>
</Catalog>
Arquivo XML de exemplo: Ordens de compra
consolidadas
23/10/2019 • 2 minutes to read • Edit Online
O arquivo XML a seguir é usado em vários exemplos na documentação do LINQ to XML. Este arquivo é um
conjunto de pedidos de compra com várias formas diferentes de empresas. Os pedidos de compra de cada
empresa em namespaces são separadas.
ConsolidatedPurchaseOrders.xml
<?xml version="1.0"?>
<PurchaseOrders xmlns="www.contoso.com">
<PurchaseOrder
PurchaseOrderNumber="99503"
OrderDate="1999-10-20">
<Address Type="Shipping">
<Name>Ellen Adams</Name>
<Street>123 Maple Street</Street>
<City>Mill Valley</City>
<State>CA</State>
<Zip>10999</Zip>
<Country>USA</Country>
</Address>
<Address Type="Billing">
<Name>Tai Yee</Name>
<Street>8 Oak Avenue</Street>
<City>Old Town</City>
<State>PA</State>
<Zip>95819</Zip>
<Country>USA</Country>
</Address>
<DeliveryNotes>Please leave packages in shed by driveway.</DeliveryNotes>
<Items>
<Item PartNumber="872-AA">
<ProductName>Lawnmower</ProductName>
<Quantity>1</Quantity>
<USPrice>148.95</USPrice>
<Comment>Confirm this is electric</Comment>
</Item>
<Item PartNumber="926-AA">
<ProductName>Baby Monitor</ProductName>
<Quantity>2</Quantity>
<USPrice>39.98</USPrice>
<ShipDate>1999-05-21</ShipDate>
</Item>
</Items>
</PurchaseOrder>
<PurchaseOrder PurchaseOrderNumber="99505" OrderDate="1999-10-22">
<Address Type="Shipping">
<Name>Cristian Osorio</Name>
<Street>456 Main Street</Street>
<City>Buffalo</City>
<State>NY</State>
<Zip>98112</Zip>
<Country>USA</Country>
</Address>
<Address Type="Billing">
<Name>Cristian Osorio</Name>
<Street>456 Main Street</Street>
<City>Buffalo</City>
<City>Buffalo</City>
<State>NY</State>
<Zip>98112</Zip>
<Country>USA</Country>
</Address>
<DeliveryNotes>Please notify by email before shipping.</DeliveryNotes>
<Items>
<Item PartNumber="456-NM">
<ProductName>Power Supply</ProductName>
<Quantity>1</Quantity>
<USPrice>45.99</USPrice>
</Item>
</Items>
</PurchaseOrder>
<PurchaseOrder PurchaseOrderNumber="99504" OrderDate="1999-10-22">
<Address Type="Shipping">
<Name>Jessica Arnold</Name>
<Street>4055 Madison Ave</Street>
<City>Seattle</City>
<State>WA</State>
<Zip>98112</Zip>
<Country>USA</Country>
</Address>
<Address Type="Billing">
<Name>Jessica Arnold</Name>
<Street>4055 Madison Ave</Street>
<City>Buffalo</City>
<State>NY</State>
<Zip>98112</Zip>
<Country>USA</Country>
</Address>
<DeliveryNotes>Please do not deliver on Saturday.</DeliveryNotes>
<Items>
<Item PartNumber="898-AZ">
<ProductName>Computer Keyboard</ProductName>
<Quantity>1</Quantity>
<USPrice>29.99</USPrice>
</Item>
<Item PartNumber="898-AM">
<ProductName>Wireless Mouse</ProductName>
<Quantity>1</Quantity>
<USPrice>14.99</USPrice>
</Item>
</Items>
</PurchaseOrder>
<aw:PurchaseOrder
PONumber="11223"
Date="2000-01-15"
xmlns:aw="http://www.adventure-works.com">
<aw:ShippingAddress>
<aw:Name>Chris Preston</aw:Name>
<aw:Street>123 Main St.</aw:Street>
<aw:City>Seattle</aw:City>
<aw:State>WA</aw:State>
<aw:Zip>98113</aw:Zip>
<aw:Country>USA</aw:Country>
</aw:ShippingAddress>
<aw:BillingAddress>
<aw:Name>Chris Preston</aw:Name>
<aw:Street>123 Main St.</aw:Street>
<aw:City>Seattle</aw:City>
<aw:State>WA</aw:State>
<aw:Zip>98113</aw:Zip>
<aw:Country>USA</aw:Country>
</aw:BillingAddress>
<aw:DeliveryInstructions>Ship only complete order.</aw:DeliveryInstructions>
<aw:Item PartNum="LIT-01">
<aw:ProductID>Litware Networking Card</aw:ProductID>
<aw:Qty>1</aw:Qty>
<aw:Price>20.99</aw:Price>
</aw:Item>
<aw:Item PartNum="LIT-25">
<aw:ProductID>Litware 17in LCD Monitor</aw:ProductID>
<aw:Qty>1</aw:Qty>
<aw:Price>199.99</aw:Price>
</aw:Item>
</aw:PurchaseOrder>
</PurchaseOrders>
Referência (LINQ to XML)
23/10/2019 • 2 minutes to read • Edit Online
Nesta seção
Para documentação de referência para as classes LINQ to XML, consulte System.Xml.Linq.
Para documentação de referência para os métodos de extensão que ajudam a validar árvores XML em um arquivo
XSD, consulte System.Xml.Schema.Extensions.
Para documentação de referência para os métodos de extensão que permitem avaliar consultas XPath em uma
árvore XML, consulte System.Xml.XPath.Extensions.
Consulte também
LINQ to XML (C#)
LINQ to ADO.NET (página do portal)
23/10/2019 • 3 minutes to read • Edit Online
O LINQ to ADO.NET permite que você realize consultas em qualquer objeto enumerável em ADO.NET usando o
modelo de programação LINQ (Consulta Integrada à Linguagem).
NOTE
A documentação do LINQ to ADO.NET está localizada na seção ADO.NET do SDK do .NET Framework: LINQ e ADO.NET.
Há três tecnologias separadas do LINQ (Consulta Integrada à Linguagem) do ADO.NET: LINQ to DataSet, LINQ
to SQL e LINQ to Entities. O LINQ to DataSet fornece consultas mais sofisticadas e otimizadas do que o DataSet,
o LINQ to SQL permite que você consulte diretamente os esquemas de banco de dados do SQL Server e o LINQ
to Entities permite que você consulte um Modelo de Dados de Entidade.
LINQ to DataSet
O DataSet é um dos componentes mais amplamente usados em ADO.NET e é um elemento fundamental do
modelo de programação desconectada no qual o ADO.NET se baseia. No entanto, apesar dessa importância, o
DataSet limitou os recursos de consulta.
O LINQ to DataSet permite que você crie recursos mais sofisticados de consulta no DataSet usando a mesma
funcionalidade de consulta que está disponível para muitas outras fontes de dados.
Para obter mais informações, consulte LINQ to DataSet.
LINQ to SQL
O LINQ to SQL fornece uma infraestrutura em tempo de execução para gerenciar dados relacionais como objetos.
No LINQ to SQL, o modelo de dados de um banco de dados relacional é mapeado para um modelo de objeto
expresso na linguagem de programação do desenvolvedor. Quando você executa o aplicativo, o LINQ to SQL
converte consultas integradas da linguagem no modelo de objeto no SQL e as envia para o banco de dados para
execução. Quando o banco de dados retorna os resultados, o LINQ to SQL os converte em objetos que podem ser
manipulados.
LINQ to SQL inclui suporte para procedimentos armazenados e funções definidas pelo usuário no banco de dados
e para herança no modelo de objeto.
Para obter mais informações, consulte LINQ to SQL.
LINQ to Entities
Por meio do Modelo de Dados de Entidade, os dados relacionais são expostos como objetos no ambiente .NET.
Isso torna a camada do objeto um destino ideal para o suporte do LINQ permitindo que os desenvolvedores
formulem consultas no banco de dados na linguagem usada para criar a lógica de negócios. Essa funcionalidade é
conhecida como LINQ to Entities. Consulte LINQ to Entities para obter mais informações.
Consulte também
LINQ e ADO.NET
LINQ (consulta integrada à linguagem) (C#)
Habilitando uma fonte de dados para consulta LINQ
23/10/2019 • 7 minutes to read • Edit Online
Há várias formas de estender o LINQ para permitir que qualquer fonte de dados seja consultada no padrão LINQ.
A fonte de dados pode ser uma estrutura de dados, um serviço Web, um sistema de arquivos ou um banco de
dados, apenas para citar algumas opções. O padrão LINQ torna fácil para os clientes consultarem uma fonte de
dados para a qual a consulta do LINQ está habilitada, porque a sintaxe e o padrão de consulta não mudam. As
formas nas quais o LINQ pode ser estendido para essas fontes de dados incluem as seguintes:
Implementação da interface IEnumerable<T> em um tipo para habilitar LINQ para consultas de Objetos
desse tipo.
Criação de métodos de operador de consulta padrão, como Where e Select que estendem um tipo, para
habilitar consultas personalizadas de LINQ desse tipo.
Criação de um provedor para sua fonte de dados que implemente a interface IQueryable<T>. Um provedor
que implementa a interface recebe consultas de LINQ na forma de árvores de expressões, as quais ele pode
executar de forma personalizada, por exemplo, remotamente.
Criar um provedor para sua fonte de dados que se beneficia de uma tecnologia existente do LINQ. Tal
provedor habilitaria não apenas a consulta, mas também operações de inserção, atualização e exclusão e
mapeamento para tipos definidos pelo usuário.
Este tópico aborda essas opções.
Consulte também
IQueryable<T>
IEnumerable<T>
Enumerable
Visão geral de operadores de consulta padrão (C#)
LINQ to Objects (C#)
Suporte de ferramentas e do IDE do Visual Studio
para LINQ (C#)
23/10/2019 • 2 minutes to read • Edit Online
O IDE (ambiente de desenvolvimento integrado) do Visual Studio fornece os seguintes recursos que dão suporte
ao desenvolvimento de aplicativos LINQ:
Consulte também
LINQ (consulta integrada à linguagem) (C#)
Programação orientada a objeto (C#)
04/11/2019 • 18 minutes to read • Edit Online
Classes e objetos
Os termos classe e objeto às vezes são usados de forma intercambiável, mas, na verdade, as classes descrevem o
tipo dos objetos, enquanto os objetos são instâncias utilizáveis das classes. Sendo assim, o ato de criar um objeto é
chamado de instanciação. Usando a analogia da uma planta, uma classe é a planta e um objeto é a construção
feita com base naquela planta.
Para definir uma classe:
class SampleClass
{
}
O C# também oferece uma versão leve das classes chamada de estruturas, que são úteis quando você precisa criar
uma matriz grande de objetos e não quer consumir muita memória para isso.
Para definir uma estrutura:
struct SampleStruct
{
}
class SampleClass
{
public string sampleField;
}
As propriedades têm procedimentos de obter e definir, que oferecem mais controle sobre como os valores são
definidos ou retornados.
O C# permite criar um campo particular para armazenar o valor da propriedade ou usar as chamadas propriedade
autoimplementada que criam esse campo automaticamente em segundo plano e fornecem a lógica básica para os
procedimentos de propriedade.
Para definir uma propriedade autoimplementada:
class SampleClass
{
public int SampleProperty { get; set; }
}
Se você precisar executar operações adicionais para ler e gravar o valor da propriedade, defina um campo para
armazenar o valor da propriedade e forneça a lógica básica para armazenar e recuperá-la:
class SampleClass
{
private int _sample;
public int Sample
{
// Return the value stored in a field.
get { return _sample; }
// Store the value in the field.
set { _sample = value; }
}
}
A maioria das propriedades têm métodos ou procedimentos para definir e obter o valor da propriedade. No
entanto, você pode criar propriedades somente leitura ou somente gravação para impedir que elas sejam
modificadas ou lidas. No C#, é possível omitir o método de propriedade get ou set . No entanto, propriedades
autoimplementadas não podem ser somente leitura ou somente gravação.
Para obter mais informações, consulte:
get
set
Métodos
Um método é uma ação que um objeto pode executar.
Para definir um método de uma classe:
class SampleClass
{
public int sampleMethod(string sampleParam)
{
// Insert code here
}
}
Uma classe pode ter várias implementações ou sobrecargas, do mesmo método que diferem quanto ao número
de parâmetros ou tipos de parâmetro.
Para sobrecarregar um método:
Na maioria dos casos, você declara um método dentro de uma definição de classe. No entanto, o C# também dá
suporte a métodos de extensão que permitem adicionar métodos a uma classe existente fora da definição real da
classe.
Para obter mais informações, consulte:
Métodos
Métodos de Extensão
Construtores
Construtores são métodos de classe que são executados automaticamente quando um objeto de um determinado
tipo é criado. Os construtores normalmente inicializam os membros de dados do novo objeto. Um construtor
pode ser executado apenas uma vez quando uma classe é criada. Além disso, o código no construtor sempre é
executado antes de qualquer outro código em uma classe. No entanto, é possível criar várias sobrecargas de
construtor da mesma forma que é feita para qualquer outro método.
Para definir um construtor para uma classe:
class Container
{
class Nested
{
// Add code here.
}
}
Para criar uma instância da classe aninhada, use o nome da classe de contêiner seguido pelo ponto e, em seguida,
seguido pelo nome da classe aninhada:
protected internal O tipo ou membro pode ser acessado por qualquer código no
mesmo assembly ou por qualquer classe derivada em outro
assembly.
private protected O tipo ou membro pode ser acessado pelo código na mesma
classe ou em uma classe derivada no assembly da classe base.
Após instanciar uma classe, você pode atribuir valores às propriedades e campos da instância e invocar métodos
da classe.
Para atribuir valores a propriedades durante o processo de instanciação de classe, use os inicializadores de objeto:
Para acessar o membro estático, use o nome da classe sem criar um objeto dessa classe:
Console.WriteLine(SampleClass.SampleString);
Classes estáticas em C# têm apenas membros estáticos e não podem ser instanciadas. Membros estáticos
também não podem acessar propriedades, métodos ou campos não estáticos
Para obter mais informações, consulte static.
Tipos anônimos
Os tipos anônimos permitem criar objetos sem escrever uma definição de classe para o tipo de dados. Em vez
disso, o compilador gera uma classe para você. A classe não tem nenhum nome utilizável e contém as
propriedades que você especificar ao declarar o objeto.
Para criar uma instância de um tipo anônimo:
Herança
A herança permite que você crie uma nova classe que reutiliza, estende e modifica o comportamento definido em
outras classes. A classe cujos membros são herdados é chamada classe base e a classe que herda esses membros
é chamada classe derivada. No entanto, todas as classes em C# herdam implicitamente da classe Object que dá
suporte à hierarquia de classes do .NET e fornece serviços de nível baixo para todas as classes.
NOTE
O C# não dá suporte a várias heranças. Ou seja, você pode especificar apenas uma classe base para uma classe derivada.
class DerivedClass:BaseClass {}
Por padrão, todas as classes podem ser herdadas. No entanto, é possível especificar se uma classe não deve ser
usada como classe base ou criar uma classe que possa ser usada apenas como classe base.
Para especificar que uma classe não pode ser usada como classe base:
Para especificar que uma classe pode ser usada apenas como classe base e não pode ser instanciada:
public abstract class B { }
MODIFICADOR DE C# DEFINIÇÃO
Interfaces
Interfaces, como classes, definem um conjunto de propriedades, métodos e eventos. Mas, diferente das classes, as
interfaces não fornecem implementação. Elas são implementadas por classes e definidas como entidades
separadas das classes. Uma interface representa um contrato, no sentido em que uma classe que implementa uma
interface deve implementar todos os aspectos da interface exatamente como ela está definida.
Para definir uma interface:
interface ISampleInterface
{
void doSomething();
}
Genéricos
Classes, estruturas, interfaces e métodos do .NET Framework podem incluir parâmetros de tipo que definem tipos
de objetos que podem armazenar ou usar. O exemplo mais comum dos genéricos é uma coleção, em que você
pode especificar o tipo dos objeto a serem armazenados em uma coleção.
Para definir uma classe genérica:
Delegados
Um delegado é um tipo que define uma assinatura de método e pode fornecer uma referência a qualquer método
com uma assinatura compatível. Você pode invocar (ou chamar) o método através do delegado. Delegados são
usados para passar métodos como argumentos a outros métodos.
NOTE
Os manipuladores de eventos nada mais são do que métodos chamados por meio de delegados. Para obter mais
informações sobre como usar delegados na manipulação de eventos, consulte Eventos.
Para criar uma referência a um método que corresponde à assinatura especificada pelo delegado:
class SampleClass
{
// Method that matches the SampleDelegate signature.
public static void sampleMethod(string message)
{
// Add code here.
}
// Method that instantiates the delegate.
void SampleDelegate()
{
SampleDelegate sd = sampleMethod;
sd("Sample string");
}
}
Consulte também
Guia de Programação em C#
Reflexão (C#)
23/10/2019 • 2 minutes to read • Edit Online
A reflexão fornece objetos (do tipo Type) que descrevem assemblies, módulos e tipos. É possível usar a reflexão
para criar dinamicamente uma instância de um tipo, associar o tipo a um objeto existente ou obter o tipo de um
objeto existente e invocar seus métodos ou acessar suas propriedades e campos. Se você estiver usando
atributos em seu código, a reflexão permite acessá-los. Para obter mais informações, consulte Atributos.
Veja um exemplo simples de reflexão usando o método estático GetType – herdado por todos os tipos da classe
base Object – para obter o tipo de uma variável:
A saída é:
System.Int32
O exemplo a seguir usa a reflexão para obter o nome completo do assembly carregado.
A saída é:
mscorlib, Version=2.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089
NOTE
As palavras-chave de C# protected e internal não têm significado no IL e não são usadas nas APIs de reflexão. Os
termos correspondentes na IL são Família e Assembly. Para identificar um método internal usando a reflexão, use a
propriedade IsAssembly. Para identificar um método protected internal , use o IsFamilyOrAssembly.
Seções relacionadas
Para saber mais:
Reflexão
Exibindo informações de tipo
Reflexão e tipos genéricos
System.Reflection.Emit
Recuperando informações armazenadas em atributos
Consulte também
Guia de Programação em C#
Assemblies no .NET
Serialização (C#)
23/10/2019 • 7 minutes to read • Edit Online
A serialização é o processo de converter um objeto em um fluxo de bytes para armazenar o objeto ou transmiti-lo
para a memória, um banco de dados ou um arquivo. Sua finalidade principal é salvar o estado de um objeto para
recriá-lo quando necessário. O processo inverso é chamado desserialização.
O objeto é serializado em um fluxo, que transporta não apenas os dados, mas informações sobre o tipo de objeto,
como sua versão, cultura e nome do assembly. Desse fluxo, ele pode ser armazenado em um banco de dados, um
arquivo ou uma memória.
Usos para serialização
A serialização permite que o desenvolvedor salve o estado de um objeto e recrie conforme necessário, fornecendo
o armazenamento dos objetos, bem como a troca de dados. Pela serialização, um desenvolvedor pode executar
ações como enviar o objeto para um aplicativo remoto por meio de um serviço Web, passando um objeto de um
domínio para outro, passando um objeto por um firewall como uma cadeia de caracteres XML ou mantendo a
segurança ou as informações específicas de usuários entre aplicativos.
Tornando um objeto serializável
Para serializar um objeto, é necessário que ele esteja serializado, um fluxo contenha o objeto serializado e um
Formatter. O System.Runtime.Serialization contém as classes necessárias para serializar e desserializar objetos.
Aplique o atributo SerializableAttribute a um tipo para indicar que as instâncias desse tipo podem ser serializadas.
Uma exceção será gerada se você tentar serializar, mas o tipo não terá o atributo SerializableAttribute.
Se não desejar que um campo em sua classe seja serializável, aplique o atributo NonSerializedAttribute. Se um
campo de um tipo serializável contiver um ponteiro, um identificador ou outra estrutura de dados que é específica
de um determinado ambiente e o campo não puder ser reconstituído em um ambiente diferente, será necessário
torná-lo não serializável.
Se uma classe serializada contiver referências a objetos de outras classes que estão marcadas como
SerializableAttribute, esses objetos também serão serializados.
Serialização de designer
A serialização de designer é um formulário especial de serialização que envolve o tipo de persistência do objeto
associado a ferramentas de desenvolvimento. A serialização de designer é o processo de conversão de um grafo
do objeto em um arquivo de origem que pode, posteriormente, ser usado para recuperar o grafo do objeto. Um
arquivo de origem pode conter código, marcação ou até mesmo informações de tabela do SQL.
Este exemplo grava o objeto de uma classe para um arquivo XML usando a classe XmlSerializer.
Exemplo
public class XMLWrite
{
writer.Serialize(file, overview);
file.Close();
}
}
Compilando o código
A classe precisa ter um construtor público sem parâmetros.
Programação robusta
As seguintes condições podem causar uma exceção:
A classe que está sendo serializada não tem um construtor público sem parâmetros.
O arquivo existe e é somente leitura (IOException).
O caminho é muito longo (PathTooLongException).
O disco está cheio (IOException).
Consulte também
StreamWriter
Como: Ler dados de objeto de um arquivo XML (C#)
Serialização (C#)
Como: Ler dados de objeto de um arquivo XML (C#)
23/10/2019 • 2 minutes to read • Edit Online
Este exemplo lê dados de objeto que foram previamente gravados em um arquivo XML usando a classe
XmlSerializer.
Exemplo
public class Book
{
public String title;
}
Console.WriteLine(overview.title);
Compilando o código
Substitua o nome de arquivo "c:\temp\SerializationOverview.xml" pelo nome do arquivo que contém os dados
serializados. Para obter mais informações sobre como serializar dados, confira Como: Gravar dados de objeto em
um arquivo XML (C#).
A classe deve ter um construtor público sem parâmetros.
Somente propriedades e campos públicos são desserializados.
Programação robusta
As seguintes condições podem causar uma exceção:
A classe que está sendo serializada não tem um construtor público sem parâmetros.
Os dados no arquivo não representam dados da classe a ser desserializada.
O arquivo não existe (IOException).
Consulte também
StreamWriter
Como: Gravar dados de objeto em um arquivo XML (C#)
Serialização (C#)
Guia de Programação em C#
Passo a passo: persistir um objeto usando o C#
23/10/2019 • 8 minutes to read • Edit Online
Você pode usar a serialização para manter os dados de um objeto entre instâncias, o que permite armazenar
valores e recuperá-los na próxima vez que o objeto for instanciado.
Neste passo a passo, você criará um objeto Loan básico e persistirá seus dados em um arquivo. Em seguida, você
recuperará os dados do arquivo quando recriar o objeto.
IMPORTANT
Este exemplo criará um novo arquivo se o arquivo ainda não existir. Se um aplicativo precisar criar um arquivo, esse aplicativo
precisará ter a permissão Create para a pasta. Permissões são definidas usando listas de controle de acesso. Se o arquivo já
existir, o aplicativo precisará somente da permissão Write , que é uma permissão menor. Sempre que possível, é mais seguro
criar o arquivo durante a implantação e somente conceder permissões Read a um único arquivo (em vez das permissões
Create para uma pasta). Além disso, é mais seguro gravar dados em pastas de usuário do que na pasta raiz ou na pasta
Arquivos de Programas.
IMPORTANT
Este exemplo armazena dados em um arquivo de formato binário. Esses formatos não devem ser usados para dados
confidenciais, como senhas ou informações de cartão de crédito.
Pré-requisitos
Para criar e executar, instale o SDK do .NET Core.
Instale seu editor de código favorito, caso ainda não tenha um.
TIP
Precisa instalar um editor de código? Experimente o Visual Studio!
[field:NonSerialized()]
public DateTime TimeLastLoaded { get; set; }
[field: NonSerialized()]
public event System.ComponentModel.PropertyChangedEventHandler PropertyChanged;
Adicione um manipulador de eventos ao evento PropertyChanged e algumas linhas para modificar o objeto Loan e
exibir as alterações. Veja as adições no seguinte código:
A execução desse aplicativo repetidamente grava os mesmos valores. Um novo objeto Loan é criado sempre que o
programa é executado. No mundo real, as taxas de juros mudam periodicamente, mas não necessariamente toda
vez que o aplicativo for executado. Código de serialização significa preservar a taxa de juros mais recente entre
instâncias do aplicativo. Na próxima etapa, você fará exatamente isso adicionando a serialização à classe Loan.
[Serializable()]
O SerializableAttribute informa ao compilador que tudo na classe pode ser persistido em um arquivo. Como o
evento PropertyChanged não representa a parte do grafo do objeto que deve ser armazenada, ele não deve ser
serializado. Ao fazer isso, todos os objetos anexados a esse evento serão serializados. Adicione o
NonSerializedAttribute à declaração de campo do manipulador de eventos PropertyChanged .
[field: NonSerialized()]
public event System.ComponentModel.PropertyChangedEventHandler PropertyChanged;
Do C# 7.3 em diante, você pode anexar atributos ao campo de suporte de uma propriedade autoimplementada
usando o valor de destino field . O seguinte código adiciona uma propriedade TimeLastLoaded e a marca como
não serializável:
[field:NonSerialized()]
public DateTime TimeLastLoaded { get; set; }
A etapa seguinte é adicionar o código de serialização ao aplicativo LoanApp. Para serializar a classe e gravá-la em
um arquivo, use os namespaces System.IO e System.Runtime.Serialization.Formatters.Binary. Para evitar a
digitação dos nomes totalmente qualificados, você pode adicionar referências aos namespaces necessários,
conforme mostrado no seguinte código:
using System.IO;
using System.Runtime.Serialization.Formatters.Binary;
A próxima etapa é adicionar código para desserializar o objeto do arquivo quando o objeto for criado. Adicione
uma constante à classe do nome de arquivo dos dados serializados, conforme mostrado no seguinte código:
Em seguida, adicione o seguinte código após a linha que cria o objeto TestLoan :
if (File.Exists(FileName))
{
Console.WriteLine("Reading saved file");
Stream openFileStream = File.OpenRead(FileName);
BinaryFormatter deserializer = new BinaryFormatter();
TestLoan = (Loan)deserializer.Deserialize(openFileStream);
TestLoan.TimeLastLoaded = DateTime.Now;
openFileStream.Close();
}
Primeiro, é necessário verificar se o arquivo existe. Se ele existir, crie uma classe Stream para ler o arquivo binário e
uma classe BinaryFormatter para converter o arquivo. Você também precisa converter do tipo de fluxo para o tipo
de objeto Loan.
Em seguida, é necessário adicionar um código para serializar a classe em um arquivo. Adicione o seguinte código
após o código existente no método Main :
Neste ponto, você pode compilar e executar o aplicativo novamente. Na primeira vez em que ele é executado,
observe que as taxas de juros começam em 7,5 e, em seguida, são alteradas para 7,1. Feche o aplicativo e execute-
o novamente. Agora, o aplicativo imprime a mensagem indicando que ele leu o arquivo salvo e a taxa de juros é
7,1, mesmo antes do código que a altera.
Consulte também
Serialização (C#)
Guia de Programação em C#
Instruções, expressões e operadores (Guia de
Programação em C#)
23/10/2019 • 2 minutes to read • Edit Online
O código C# que compõe um aplicativo consiste em instruções compostas por palavras-chave, expressões e
operadores. Esta seção contém informações sobre esses elementos fundamentais de um programa C#.
Para obter mais informações, consulte:
Instruções
Expressões
Membros de expressão incorporada
Funções Anônimas
Comparações de igualdade
Especificação da Linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a fonte
definitiva para a sintaxe e o uso de C#.
Consulte também
Guia de Programação em C#
Transmissões e conversões de tipo
Instruções (Guia de Programação em C#)
23/10/2019 • 10 minutes to read • Edit Online
As ações que usa um programa executa são expressas em instruções. Ações comuns incluem declarar variáveis,
atribuir valores, chamar métodos, fazer loops pelas coleções e ramificar para um ou para outro bloco de código,
dependendo de uma determinada condição. A ordem na qual as instruções são executadas em um programa é
chamada de fluxo de controle ou fluxo de execução. O fluxo de controle pode variar sempre que um programa é
executado, dependendo de como o programa reage às entradas que recebe em tempo de execução.
Uma instrução pode consistir em uma única linha de código que termina em um ponto e vírgula ou uma série de
instruções de uma linha em um bloco. Um bloco de instrução é colocado entre colchetes {} e pode conter blocos
aninhados. O código a seguir mostra dois exemplos de instruções de linha única, bem como um bloco de
instrução de várias linhas:
// Assignment statement.
counter = 1;
Instruções para tratamento de exceções Instruções para tratamento de exceções permitem que você
se recupere normalmente de condições excepcionais que
ocorrem em tempo de execução. Para mais informações,
consulte os seguintes tópicos:
Instruções de declaração
O código a seguir mostra exemplos de declarações de variável com e sem uma atribuição inicial e uma declaração
de constante com a inicialização necessária.
Instruções de expressão
O código a seguir mostra exemplos de instruções de expressão, incluindo a atribuição, a criação de objeto com a
atribuição e a invocação de método.
// Expression statement (assignment).
area = 3.14 * (radius * radius);
A instrução vazia
Os exemplos a seguir mostram dois usos de uma instrução vazia:
void ProcessMessages()
{
while (ProcessMessage())
; // Statement needed here.
}
void F()
{
//...
if (done) goto exit;
//...
exit:
; // Statement needed here.
}
Instruções inseridas
Algumas instruções, inclusive do, while, for e foreach, sempre têm uma instrução inserida que as segue. Essa
instrução inserida pode ser uma instrução única ou várias instruções colocadas entre colchetes {} em um bloco de
instrução. Até mesmo instruções inseridas de uma única linha podem ser colocadas entre colchetes {}, conforme
mostrado no seguinte exemplo:
// Not recommended.
foreach (string s in System.IO.Directory.GetDirectories(
System.Environment.CurrentDirectory))
System.Console.WriteLine(s);
Uma instrução inserida que não está entre colchetes {} não pode ser uma instrução de declaração ou uma
instrução rotulada. Isso é mostrado no exemplo a seguir:
if(pointB == true)
//Error CS1023:
int radius = 5;
Coloque a instrução inserida em um bloco para corrigir o erro:
if (b == true)
{
// OK:
System.DateTime d = System.DateTime.Now;
System.Console.WriteLine(d.ToLongDateString());
}
}
return "Not found.";
Instruções inacessíveis
Se o compilador determinar que o fluxo de controle nunca pode atingir uma determinada instrução em nenhuma
circunstância, ele produzirá o aviso CS0162, conforme mostrado no exemplo a seguir:
Especificação da linguagem C#
Para saber mais, confira a seção Instruções da Especificação da linguagem C#.
Consulte também
Guia de Programação em C#
Palavras-chave de instrução
Expressões
Expressões (Guia de Programação em C#)
04/11/2019 • 9 minutes to read • Edit Online
Uma expressão é uma sequência de um ou mais operandos e zero ou mais operadores que podem ser avaliados
como um valor, objeto, método ou namespace único. As expressões podem consistir de um valor literal, uma
invocação de método, um operador e seus operandos ou um nome simples. Os nomes simples podem ser o
nome de uma variável, um membro de tipo, um parâmetro de método, um namespace ou um tipo.
As expressões podem usar operadores que, por sua vez, usam outras expressões como parâmetros ou chamadas
de métodos cujos parâmetros são, por sua vez, outras chamadas de métodos. Assim, as expressões podem variar
de “simples” a “muito complexas”. Estes são dois exemplos de expressões:
((x < 10) && ( x > 5)) || ((x > 20) && (x < 25));
System.Convert.ToInt32("35");
Valores de expressão
Na maioria dos contextos em que as expressões são usadas, por exemplo, em instruções ou parâmetros de
método, espera-se que a expressão seja avaliada como algum valor. Se x e y são inteiros, a expressão x + y será
avaliada como um valor numérico. A expressão new MyClass() é avaliada como uma referência a uma nova
instância de uma classe MyClass . A expressão myClass.ToString() é avaliada como uma cadeia de caracteres,
porque esse é o tipo de retorno do método. No entanto, embora um nome de namespace seja classificado como
uma expressão, ele não é avaliado como um valor e, portanto, nunca pode ser o resultado final de qualquer
expressão. Não é possível passar um nome de namespace para um parâmetro de método, usá-lo em uma nova
expressão ou atribuí-lo a uma variável. Ele poderá ser usado somente como uma subexpressão em uma
expressão maior. Isso também se aplica a tipos (diferentes dos objetos System.Type), a nomes de grupos de
métodos (diferentes de métodos específicos) e aos acessadores de eventos add e remove.
Cada valor tem um tipo associado. Por exemplo, se x e y forem variáveis do tipo int , o valor da expressão
x + y também será tipado como int . Se o valor for atribuído a uma variável de um tipo diferente ou se x e y
forem tipos diferentes, as regras de conversão de tipo serão aplicadas. Para obter mais informações sobre como
essas conversões funcionam, consulte Conversões e Conversões de Tipo.
Estouros
Expressões numéricas podem causar estouros se o valor for maior que o valor máximo do tipo de valor. Para
obter mais informações, consulte marcado e desmarcado e a seção de conversões numéricas explícitas do artigo
embutir conversões numéricas .
// Expression statements.
int i = 5;
string s = "Hello World";
int num = 5;
System.Console.WriteLine(num); // Output: 5
num = 6;
System.Console.WriteLine(num); // Output: 6
Expressões de invocação
No exemplo de código a seguir, a chamada para DoWork é uma expressão de invocação.
DoWork();
Uma invocação de método requer o nome do método, como um nome, conforme o exemplo anterior ou como
resultado de outra expressão, seguido de parênteses e quaisquer parâmetros de método. Para saber mais, veja
Métodos. Uma invocação de delegado usa o nome de um delegado e dos parâmetros de método entre
parênteses. Para obter mais informações, consulte Delegados. Invocações de método e de delegados são
avaliadas como o valor retornado do método se o método retornar um valor. Métodos que retornam nulo não
podem ser usados no lugar de um valor em uma expressão.
Expressões de consulta
As mesmas regras para expressões em geral se aplicam a expressões de consulta. Para saber mais, confira LINQ.
Expressões lambda
As expressões lambda representam "métodos embutidos" que não têm nome, mas podem ter parâmetros de
entrada e várias instruções. Elas são usadas amplamente no LINQ para passar argumentos para métodos. As
expressões lambda são compiladas para delegados ou árvores de expressão, dependendo do contexto no qual
elas são usadas. Para obter mais informações, consulte Expressões Lambda.
Árvores de expressão
As árvores de expressão habilitam expressões a serem representadas como estruturas de dados. Elas são usadas
amplamente por provedores LINQ para mover expressões de consulta para o código significativo em outro
contexto, como um banco de dados SQL. Para obter mais informações, confira Árvores de expressão (C#).
Especificação da linguagem C#
Para saber mais, confira a seção Expressões da Especificação da linguagem C#.
Consulte também
Guia de Programação em C#
Operadores
Métodos
Delegados
Tipos
LINQ
Membros aptos para expressão (Guia de
Programação em C#)
31/10/2019 • 6 minutes to read • Edit Online
As definições de corpo da expressão permitem que você forneça uma implementação de um membro em uma
forma bastante concisa e legível. Você pode usar uma definição de corpo da expressão sempre que a lógica para
qualquer membro com suporte, como um método ou propriedade, consiste em uma única expressão. Uma
definição de corpo da expressão tem a seguinte sintaxe geral:
Método C# 6
Property C# 7.0
Construtor C# 7.0
Finalizador C# 7.0
Indexador C# 7.0
Métodos
Um método apto para expressão consiste em uma única expressão que retorna um valor cujo tipo corresponde
ao tipo de retorno do método, ou, para métodos que retornam void , que executam uma operação. Por exemplo,
os tipos que substituem o método ToString normalmente incluem uma única expressão que retorna a
representação da cadeia de caracteres do objeto atual.
O exemplo a seguir define uma classe Person que substitui o método ToString por uma definição de corpo da
expressão. Ele também define um método DisplayName que exibe um nome para o console. Observe que a
palavra-chave return não é usada na definição de corpo da expressão ToString .
using System;
class Example
{
static void Main()
{
Person p = new Person("Mandy", "Dejesus");
Console.WriteLine(p);
p.DisplayName();
}
}
O exemplo a seguir define uma classe Location cuja propriedade somente leitura Name é implementada como
uma definição de corpo da expressão que retorna o valor do campo locationName particular:
Para obter mais informações sobre as propriedades, confira Propriedades (Guia de Programação em C#).
Propriedades
Começando no C# 7.0, você pode usar as definições de corpo da expressão para implementar a propriedade
get e os acessadores set . O exemplo a seguir demonstra como fazer isso:
public class Location
{
private string locationName;
Para obter mais informações sobre as propriedades, confira Propriedades (Guia de Programação em C#).
Construtores
Uma definição de corpo da expressão para um construtor normalmente consiste em uma expressão de atribuição
simples ou uma chamada de método que manipula os argumentos do construtor ou inicializa o estado da
instância.
O exemplo a seguir define uma classe Location cujo construtor tem um único parâmetro de cadeia de caracteres
chamado nome. A definição de corpo da expressão atribui o argumento à propriedade Name .
Finalizadores
Uma definição de corpo da expressão para um finalizador normalmente contém instruções de limpeza, como
instruções que liberam recursos não gerenciados.
O exemplo a seguir define um finalizador que usa uma definição de corpo da expressão para indicar que o
finalizador foi chamado.
using System;
using System;
using System.Collections.Generic;
Uma função anônima é uma instrução ou expressão "embutida" que pode ser usada em qualquer local em que
um tipo delegado é esperado. Você pode usá-la para inicializar um delegado nomeado ou passá-la em vez de um
tipo delegado nomeado como um parâmetro de método.
Você pode usar uma expressão lambda ou um método anônimo para criar uma função anônima.
Recomendamos o uso de expressões lambda já que elas fornecem uma maneira mais concisa e expressiva de
gravar código embutido. Ao contrário dos métodos anônimos, alguns tipos de expressões lambda podem ser
convertidos nos tipos de árvore de expressão.
A evolução de delegados em C#
No C# 1.0, você criava uma instância de um delegado, inicializando-a explicitamente com um método que era
definido em outro local no código. O C# 2.0 introduziu o conceito de métodos anônimos como uma maneira de
escrever blocos de instrução sem nome embutidos, que podem ser executados em uma invocação de delegado.
O C# 3.0 introduziu as expressões lambda, que são semelhantes ao conceito dos métodos anônimos, mas mais
expressivas e concisas. Essas duas funcionalidades são conhecidas coletivamente como funções anônimas. Em
geral, aplicativos que se destinam à versão 3.5 e posteriores do .NET Framework devem usar as expressões
lambda.
O exemplo a seguir demonstra a evolução da criação de delegado, desde o C# 1.0 até o C# 3.0:
class Test
{
delegate void TestDelegate(string s);
static void M(string s)
{
Console.WriteLine(s);
}
Especificação da linguagem C#
Para obter mais informações, confira a seção Expressões de função anônima da Especificação da linguagem C#.
Consulte também
Instruções, expressões e operadores
Expressões Lambda
Delegados
Árvores de expressão (C#)
Expressões lambda (Guia de Programação em
C#)
04/11/2019 • 19 minutes to read • Edit Online
Uma expressão lambda é uma expressão de uma das duas seguintes formas:
Expressão lambda que tem uma expressão como corpo:
Use o operador de declaração lambda => para separar a lista de parâmetros de lambda do corpo. Para
criar uma expressão lambda, especifique os parâmetros de entrada (se houver) no lado esquerdo do
operador lambda e uma expressão ou um bloco de instrução do outro lado.
Qualquer expressão lambda pode ser convertida para um tipo delegado. O tipo delegado no qual uma
expressão lambda pode ser convertida é definido pelos tipos de parâmetros e pelo valor retornado. Se
uma expressão lambda não retornar um valor, ela poderá ser convertida em um dos tipos delegados
Action ; caso contrário, ela poderá ser convertida em um dos tipos delegados Func . Por exemplo, uma
expressão lambda que tem dois parâmetros e não retorna nenhum valor pode ser convertida em um
delegado Action<T1,T2>. Uma expressão lambda que tem um parâmetro e retorna um valor pode ser
convertida em um delegado Func<T,TResult>. No seguinte exemplo, a expressão lambda x => x * x ,
que especifica um parâmetro chamado x e retorna o valor de x quadrado, é atribuída a uma variável
de um tipo delegado:
As expressões lambdas também podem ser convertidas nos tipos de árvore de expressão, como mostra
o seguinte exemplo:
Use expressões lambda em qualquer código que exija instâncias de tipos delegados ou árvores de
expressão, por exemplo, como um argumento ao método Task.Run(Action) para passar o código que
deve ser executado em segundo plano. Você também pode usar expressões lambda ao escrever LINQ
no C# , como mostra o exemplo a seguir:
int[] numbers = { 2, 3, 4, 5 };
var squaredNumbers = numbers.Select(x => x * x);
Console.WriteLine(string.Join(" ", squaredNumbers));
// Output:
// 4 9 16 25
Quando você usa a sintaxe baseada em método para chamar o método Enumerable.Select na classe
System.Linq.Enumerable, por exemplo, no LINQ to Objects e no LINQ to XML, o parâmetro é um tipo
delegado System.Func<T,TResult>. Quando você chama o método Queryable.Select na classe
System.Linq.Queryable, por exemplo, no LINQ to SQL, o tipo de parâmetro é um tipo de árvore de
expressão Expression<Func<TSource,TResult>> . Em ambos os casos, você pode usar a mesma expressão
lambda para especificar o valor do parâmetro. Isso faz com que as duas chamadas Select pareçam
semelhantes, embora, na verdade, o tipo de objetos criado dos lambdas seja diferente.
Lambdas de expressão
Uma expressão lambda com uma expressão no lado direito do operador => é chamada de lambda de
expressão. Os lambdas de expressão são usados amplamente na construção de árvores de expressão.
Uma expressão lambda retorna o resultado da expressão e tem o seguinte formato básico:
Os parênteses serão opcionais somente se o lambda tiver um parâmetro de entrada; caso contrário, eles
serão obrigatórios.
Especifique parâmetros de entrada zero com parênteses vazios:
Dois ou mais parâmetros de entrada são separados por vírgulas e envolvidos por parênteses:
Às vezes, é difícil ou impossível para o compilador inferir os tipos de entrada. Você pode especificar os
tipos de maneira explícita conforme mostrado neste exemplo:
Os tipos de parâmetro de entrada devem ser todos explícitos ou implícitos; caso contrário, ocorrerá o
erro CS0748 de compilador.
O corpo de um lambda de expressão pode consistir em uma chamada de método. No entanto, se você
estiver criando árvores de expressão que serão avaliadas fora contexto do .NET Common Language
Runtime, como no SQL Server, você não deverá usar chamadas de método em lambdas de expressão.
Os métodos não terão significado fora do contexto do .NET Common Language Runtime.
Lambdas de instrução
Um lambda de instrução lembra um lambda de expressão, exceto que as instruções estão incluídas entre
chaves:
(input-parameters) => { <sequence-of-statements> }
O corpo de uma instrução lambda pode consistir de qualquer número de instruções; no entanto, na
prática, normalmente não há mais de duas ou três.
Os lambdas de instrução não podem ser usados para criar árvores de expressão.
Lambdas assíncronos
Você pode facilmente criar expressões e instruções lambda que incorporem processamento assíncrono,
ao usar as palavras-chaves async e await. Por exemplo, o exemplo do Windows Forms a seguir contém
um manipulador de eventos que chama e espera um método assíncrono ExampleMethodAsync .
Você pode adicionar o mesmo manipulador de eventos ao usar um lambda assíncrono. Para adicionar
esse manipulador, adicione um modificador async antes da lista de parâmetros lambda, como mostra o
exemplo a seguir:
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
button1.Click += async (sender, e) =>
{
await ExampleMethodAsync();
textBox1.Text += "\r\nControl returned to Click event handler.\n";
};
}
Para obter mais informações sobre como criar e usar os métodos assíncronos, consulte Programação
assíncrona com async e await.
Func<(int, int, int), (int, int, int)> doubleThem = ns => (2 * ns.Item1, 2 * ns.Item2, 2 *
ns.Item3);
var numbers = (2, 3, 4);
var doubledNumbers = doubleThem(numbers);
Console.WriteLine($"The set {numbers} doubled: {doubledNumbers}");
// Output:
// The set (2, 3, 4) doubled: (4, 6, 8)
Normalmente, os campos de uma tupla são nomeados Item1 , Item2 , etc. No entanto, você pode definir
uma tupla com componentes nomeados, como faz o exemplo a seguir.
Func<(int n1, int n2, int n3), (int, int, int)> doubleThem = ns => (2 * ns.n1, 2 * ns.n2, 2 *
ns.n3);
var numbers = (2, 3, 4);
var doubledNumbers = doubleThem(numbers);
Console.WriteLine($"The set {numbers} doubled: {doubledNumbers}");
Para saber mais sobre tuplas C#, confira o artigo sobre tipos de tuplas C#.
O delegado pode ser instanciado como um Func<int, bool> , em que int é um parâmetro de entrada e
bool é o valor de retorno. O valor de retorno é sempre especificado no último parâmetro de tipo. Por
exemplo, Func<int, string, bool> define um delegado com dois parâmetros de entrada, int e string ,
e um tipo de retorno de bool . O delegado Func a seguir, quando é invocado, retornará um valor
booliano que indica se o parâmetro de entrada é ou não igual a cinco:
Você também pode fornecer uma expressão lambda quando o tipo de argumento é um
Expression<TDelegate>. Por exemplo, nos operadores de consulta padrão que são definidos no tipo
Queryable. Quando você especifica um argumento Expression<TDelegate>, o lambda é compilado em
uma árvore de expressão.
O exemplo a seguir usa o operador padrão de consulta Count:
int[] numbers = { 5, 4, 1, 3, 9, 8, 6, 7, 2, 0 };
int oddNumbers = numbers.Count(n => n % 2 == 1);
Console.WriteLine($"There are {oddNumbers} odd numbers in {string.Join(" ", numbers)}");
O compilador pode inferir o tipo de parâmetro de entrada ou você também pode especificá-lo
explicitamente. Essa expressão lambda em particular conta esses inteiros ( n ) que, quando dividida por
dois, tem um resto 1.
O exemplo a seguir gera uma sequência que contém todos os elementos da matriz numbers que
precedem o 9, porque esse é o primeiro número na sequência que não satisfaz a condição:
int[] numbers = { 5, 4, 1, 3, 9, 8, 6, 7, 2, 0 };
var firstNumbersLessThanSix = numbers.TakeWhile(n => n < 6);
Console.WriteLine(string.Join(" ", firstNumbersLessThanSix));
// Output:
// 5 4 1 3
O exemplo a seguir especifica vários parâmetros de entrada, colocando-os entre parênteses. O método
retorna todos os elementos na matriz numbers até encontrar um número cujo valor seja inferior à sua
posição ordinal na matriz:
int[] numbers = { 5, 4, 1, 3, 9, 8, 6, 7, 2, 0 };
var firstSmallNumbers = numbers.TakeWhile((n, index) => n >= index);
Console.WriteLine(string.Join(" ", firstSmallNumbers));
// Output:
// 5 4
updateCapturedLocalVariable = x =>
{
j = x;
bool result = j > input;
Console.WriteLine($"{j} is greater than {input}: {result}");
};
isEqualToCapturedLocalVariable = x => x == j;
int gameInput = 5;
game.Run(gameInput);
int anotherJ = 3;
game.updateCapturedLocalVariable(anotherJ);
Especificação da linguagem C#
Para obter mais informações, confira a seção Expressões de função anônima da Especificação da
linguagem C#.
Consulte também
Guia de Programação em C#
LINQ (Consulta Integrada à Linguagem)
Árvores de Expressão
Funções locais comparadas com expressões lambda
Expressões lambda tipadas implicitamente
Exemplos de C# do Visual Studio 2008 (veja os arquivos de exemplo de consultas LINQ e programa
XQuery)
Expressões lambda recursivas
Como usar expressões lambda em uma consulta
(Guia de Programação em C#)
04/11/2019 • 3 minutes to read • Edit Online
Você não usa expressões lambda diretamente na sintaxe da consulta, mas as usa em chamadas de método e as
expressões de consulta podem conter chamadas de método. Na verdade, algumas operações de consulta podem
ser expressas na sintaxe de método. Para obter mais informações sobre a diferença entre a sintaxe de consulta e
sintaxe de método, consulte Sintaxe de consulta e sintaxe de método em LINQ.
Exemplo
O exemplo a seguir demonstra como usar uma expressão lambda em uma consulta baseada em método usando
operador de consulta padrão Enumerable.Where. O método Where nesse exemplo tem um parâmetro de entrada
do tipo delegado Func<T,TResult>, e esse delegado utiliza um inteiro como entrada e retorna um booliano. A
expressão lambda pode ser convertida para esse delegado. Se essa fosse uma consulta LINQ to SQL que usasse o
método Queryable.Where, o tipo de parâmetro seria um Expression<Func<int,bool>> , mas a expressão lambda
teria exatamente a mesma aparência. Para saber mais sobre o tipo de expressão, confira
System.Linq.Expressions.Expression.
class SimpleLambda
{
static void Main()
{
// Data source.
int[] scores = { 90, 71, 82, 93, 75, 82 };
Exemplo
O exemplo a seguir demonstra como usar uma expressão lambda em uma chamada de método de uma expressão
de consulta. O lambda é necessário porque o operador de consulta padrão Sum não pode ser invocado usando a
sintaxe de consulta.
A consulta primeiro agrupa os alunos de acordo com o nível de ensino, conforme definido no enum GradeLevel .
Em seguida, para cada grupo, ela adiciona as pontuações totais de cada aluno. Isso requer duas operações Sum . O
Sum interno calcula a pontuação total para cada estudante e o Sum externo mantém um total acumulado
combinado para todos os alunos no grupo.
private static void TotalsByGradeLevel()
{
// This query retrieves the total scores for First Year students, Second Years, and so on.
// The outer Sum method uses a lambda in order to specify which numbers to add together.
var categories =
from student in students
group student by student.Year into studentGroup
select new { GradeLevel = studentGroup.Key, TotalScore = studentGroup.Sum(s => s.ExamScores.Sum()) };
Compilando o código
Para executar esse código, copie e cole o método no StudentClass que é fornecido em Como consultar uma
coleção de objetos e chame-o do método Main .
Consulte também
Expressões Lambda
Árvores de expressão (C#)
Comparações de igualdade (Guia de Programação
em C#)
29/11/2019 • 6 minutes to read • Edit Online
Às vezes, é necessário comparar dois valores em relação à igualdade. Em alguns casos, testa-se a igualdade de
valor, também conhecida como equivalência, o que significa que os valores contidos pelas duas variáveis são
iguais. Em outros casos, é necessário determinar se duas variáveis se referem ao mesmo objeto subjacente na
memória. Esse tipo de igualdade é chamado igualdade de referência ou identidade. Este tópico descreve esses
dois tipos de igualdade e fornece links para outros tópicos que fornecem mais informações.
Igualdade de referência
Igualdade de referência significa que as duas referências de objeto se referem ao mesmo objeto subjacente. Isso
pode ocorrer por meio de uma atribuição simples, conforme mostrado no exemplo a seguir.
using System;
class Test
{
public int Num { get; set; }
public string Str { get; set; }
// Assign b to a.
b = a;
Nesse código, dois objetos são criados, mas após a instrução de atribuição, ambas as referências se referem ao
mesmo objeto. Portanto, eles têm igualdade de referência. Use o método ReferenceEquals para determinar se
duas referências referenciam o mesmo objeto.
O conceito de igualdade de referência se aplica apenas a tipos de referência. Objetos de tipo de valor não podem
ter igualdade de referência, pois quando uma instância de um tipo de valor é atribuída a uma variável, uma cópia
do valor é gerada. Portanto, não é possível ter dois structs desconvertidos que referenciam o mesmo local na
memória. Além disso, se ReferenceEquals for usado para comparar dois tipos de valor, o resultado sempre será
false , mesmo se os valores contidos nos objetos forem idênticos. Isso ocorre porque cada variável é convertido
em uma instância de objeto separada. Para obter mais informações, consulte Como Testar a Igualdade de
Referência (Identidade).
Igualdade de valor
Igualdade de valor significa que dois objetos contêm o mesmo valor ou valores. Para tipos de valor primitivos,
como int ou bool, os testes de igualdade de valor são simples. É possível usar o operador ==, conforme mostrado
no exemplo a seguir.
int a = GetOriginalValue();
int b = GetCurrentValue();
Para a maioria dos outros tipos, o teste de igualdade de valor é mais complexo, pois é necessário entender como o
tipo o define. Para classes e structs que têm vários campos ou propriedades, a igualdade de valor geralmente é
definida para determinar que todos os campos ou propriedades tenham o mesmo valor. Por exemplo, dois objetos
Point podem ser definidos para serem equivalentes se pointA.X for igual a pointB.X e pointA.Y for igual a
pointB.Y.
No entanto, não há nenhuma exigência de que a equivalência seja baseada em todos os campos em um tipo. Ela
pode ser baseada em um subconjunto. Ao comparar tipos que não são de sua propriedade, certifique-se de que a
forma como a equivalência é definida especificamente para esse tipo foi entendida. Para obter mais informações
sobre como definir a igualdade de valor em suas próprias classes e structs, consulte Como Definir a Igualdade de
Valor para um Tipo.
Igualdade de valor para valores de ponto flutuante
As comparações de igualdade de valores de ponto flutuante (double e float) são problemáticas devido à
imprecisão da aritmética de ponto flutuante em computadores binários. Para obter mais informações, consulte os
comentários no tópico System.Double.
Tópicos relacionados
CARGO DESCRIÇÃO
Como testar a igualdade de referência (identidade) Descreve como determinar se duas variáveis têm igualdade de
referência.
Como definir a igualdade de valor para um tipo Descreve como fornecer uma definição personalizada de
igualdade de valor a um tipo.
Consulte também
Guia de Programação em C#
Como definir a igualdade de valor para um tipo
(Guia de programação em C#)
23/10/2019 • 11 minutes to read • Edit Online
Quando você define uma classe ou struct, decide se faz sentido criar uma definição personalizada de igualdade de
valor (ou equivalência) para o tipo. Normalmente, você implementa igualdade de valor quando objetos do tipo
devem ser adicionados a uma coleção de algum tipo ou quando seu objetivo principal é armazenar um conjunto
de campos ou propriedades. Você pode basear sua definição de igualdade de valor em uma comparação de todos
os campos e propriedades no tipo ou pode basear a definição em um subconjunto. Mas, em ambos os casos e em
classes e struct, sua implementação deve seguir as cinco garantias de equivalência:
1. x.Equals(x) retorna true . Isso é chamado de propriedade reflexiva.
2. x.Equals(y) retorna o mesmo valor que y.Equals(x) . Isso é chamado de propriedade simétrica.
3. se (x.Equals(y) && y.Equals(z)) retornar true , então x.Equals(z) retornará true . Isso é chamado de
propriedade transitiva.
4. Invocações sucessivas de x.Equals(y) retornam o mesmo valor, contanto que os objetos referenciados
por x e y não sejam modificados.
5. x.Equals(null) retorna false . No entanto, null.Equals(null) gera uma exceção; não obedece à regra
número dois acima.
Qualquer struct que você define já tem uma implementação padrão de igualdade de valor que ele herda da
substituição System.ValueType do método Object.Equals(Object). Essa implementação usa a reflexão para
examinar todos os campos e propriedades no tipo. Embora essa implementação produza resultados corretos, ela
é relativamente lenta em comparação com uma implementação personalizada escrita especificamente para o tipo.
Os detalhes de implementação para a igualdade de valor são diferentes para classes e struct. No entanto, as
classes e structs exigem as mesmas etapas básicas para implementar a igualdade:
1. Substitua o método virtual Object.Equals(Object). Na maioria dos casos, sua implementação de
bool Equals( object obj ) deve apenas chamar o método Equals específico do tipo que é a
implementação da interface System.IEquatable<T>. (Consulte a etapa 2.)
2. Implemente a interface System.IEquatable<T> fornecendo um método Equals específico do tipo. Isso é o
local em que a comparação de equivalência de fato é realizada. Por exemplo, você pode decidir definir a
igualdade comparando apenas um ou dois campos em seu tipo. Não lance exceções de Equals . Para
classes somente: esse método deve examinar somente os campos que são declarados na classe. Ele deve
chamar base.Equals para examinar os campos que estão na classe base. (Não faça isso se o tipo herda
diretamente de Object, pois a implementação Object de Object.Equals(Object) executa uma verificação de
igualdade de referência.)
3. Opcional, mas recomendável: Sobrecarregar os operadores == e ! =.
4. Substitua Object.GetHashCode para que os dois objetos que têm a igualdade de valor produzam o mesmo
código hash.
5. Opcional: para dar suporte às definições para “maior que” ou “menor que”, implemente a interface
IComparable<T> para seu tipo e também sobrecarregue os operadores <= e >=.
O primeiro exemplo a seguir mostra uma implementação da classe. O segundo exemplo mostra uma
implementação de struct.
Exemplo
O exemplo a seguir mostra como implementar a igualdade de valor em uma classe (tipo de referência).
namespace ValueEquality
{
using System;
class TwoDPoint : IEquatable<TwoDPoint>
{
// Readonly auto-implemented properties.
public int X { get; private set; }
public int Y { get; private set; }
class Program
{
static void Main(string[] args)
{
ThreeDPoint pointA = new ThreeDPoint(3, 4, 5);
ThreeDPoint pointB = new ThreeDPoint(3, 4, 5);
ThreeDPoint pointC = null;
int i = 5;
/* Output:
pointA.Equals(pointB) = True
pointA == pointB = True
null comparison = False
Compare to some other type = False
Two null TwoDPoints are equal: True
(pointE == pointA) = False
(pointE == pointA) = False
(pointA == pointE) = False
(pointA != pointE) = True
pointE.Equals(list[0]): False
*/
}
Exemplo
O exemplo a seguir mostra como implementar a igualdade de valor em um struct (tipo de valor):
class Program
{
static void Main(string[] args)
static void Main(string[] args)
{
TwoDPoint pointA = new TwoDPoint(3, 4);
TwoDPoint pointB = new TwoDPoint(3, 4);
int i = 5;
pointD = temp;
// True:
Console.WriteLine("pointD == (pointC = 3,4) = {0}", pointD == pointC);
/* Output:
pointA.Equals(pointB) = True
pointA == pointB = True
Object.Equals(pointA, pointB) = True
pointA.Equals(null) = False
(pointA == null) = False
(pointA != null) = True
pointA.Equals(i) = False
pointE.Equals(list[0]): True
pointA == (pointC = null) = False
pointC == pointD = True
pointA == (pointC = 3,4) = True
pointD == (pointC = 3,4) = True
*/
}
Para estruturas, a implementação padrão de Object.Equals(Object) (que é a versão substituída em
System.ValueType) executa uma verificação de igualdade de valor por meio de reflexão para comparar os valores
de cada campo no tipo. Quando um implementador substitui o método Equals virtual em uma estrutura, a
finalidade é fornecer uma maneira mais eficiente de executar a verificação de igualdade de valor e,
opcionalmente, basear a comparação em algum subconjunto dos campos ou propriedades do struct.
Os operadores == e != não podem operar em um struct a menos que o struct explicitamente os sobrecarregue.
Consulte também
Comparações de igualdade
Guia de Programação em C#
Como: testar a igualdade de referência (identidade)
(Guia de Programação em C#)
31/10/2019 • 4 minutes to read • Edit Online
Não é necessário implementar qualquer lógica personalizada para dar suporte a comparações de igualdade de
referência em seus tipos. Essa funcionalidade é fornecida para todos os tipos pelo método estático
Object.ReferenceEquals.
O exemplo a seguir mostra como determinar se duas variáveis têm igualdade de referência, que significa que elas
se referem ao mesmo objeto na memória.
O exemplo também mostra por que Object.ReferenceEquals sempre retorna false para tipos de valor e por que
você não deve usar ReferenceEquals para determinar igualdade de cadeia de caracteres.
Exemplo
namespace TestReferenceEquality
{
struct TestStruct
{
public int Num { get; private set; }
public string Name { get; private set; }
class TestClass
{
public int Num { get; set; }
public string Name { get; set; }
}
class Program
{
static void Main()
{
// Demonstrate reference equality with reference types.
#region ReferenceTypes
// Demonstrate that two value type instances never have reference equality.
#region ValueTypes
#region stringRefEquality
// Constant strings within the same assembly are always interned by the runtime.
// This means they are stored in the same location in memory. Therefore,
// the two strings have reference equality although no assignment takes place.
string strA = "Hello world!";
string strB = "Hello world!";
Console.WriteLine("ReferenceEquals(strA, strB) = {0}",
Object.ReferenceEquals(strA, strB)); // true
#endregion
/* Output:
ReferenceEquals(tcA, tcB) = False
After asignment: ReferenceEquals(tcA, tcB) = True
tcB.Name = TestClass 42 tcB.Num: 42
After asignment: ReferenceEquals(tsC, tsD) = False
ReferenceEquals(strA, strB) = True
strA = "Goodbye world!" strB = "Hello world!"
After strA changes, ReferenceEquals(strA, strB) = False
*/
A implementação de Equals na classe base universal System.Object também realiza uma verificação de
igualdade de referência, mas é melhor não usar isso, porque, se uma classe substituir o método, os resultados
poderão não ser o que você espera. O mesmo é verdadeiro para os operadores == e != . Quando eles estiverem
operando em tipos de referência, o comportamento padrão de == e != é realizar uma verificação de igualdade
de referência. No entanto, as classes derivadas podem sobrecarregar o operador para executar uma verificação de
igualdade de valor. Para minimizar o potencial de erro, será melhor usar sempre ReferenceEquals quando for
necessário determinar se os dois objetos têm igualdade de referência.
Cadeias de caracteres constantes dentro do mesmo assembly sempre são internalizadas pelo tempo de execução.
Ou seja, apenas uma instância de cada cadeia de caracteres literal única é mantida. No entanto, o tempo de
execução não garante que cadeias de caracteres criadas em tempo de execução sejam internalizadas nem garante
que duas de cadeias de caracteres constantes iguais em diferentes assemblies sejam internalizadas.
Consulte também
Comparações de igualdade
Tipos (Guia de Programação em C#)
29/11/2019 • 22 minutes to read • Edit Online
int a = 5;
int b = a + 2; //OK
// Error. Operator '+' cannot be applied to operands of type 'int' and 'bool'.
int c = a + test;
NOTE
Desenvolvedores de C e C++, observem que, em C#, bool não é conversível em int.
O compilador insere as informações de tipo no arquivo executável como metadados. O CLR (Common
Language Runtime) usa metadados em tempo de execução para garantir mais segurança de tipos quando aloca
e recupera a memória.
Especificando tipos em declarações de variável
Quando declara uma variável ou constante em um programa, você deve especificar seu tipo ou usar a palavra-
chave var para permitir que o compilador infira o tipo. O exemplo a seguir mostra algumas declarações de
variáveis que usam tipos numéricos internos e tipos complexos definidos pelo usuário:
// Declaration only:
float temperature;
string name;
MyClass myClass;
Os tipos de parâmetros de método e valores de retorno são especificados na assinatura do método. A assinatura
a seguir mostra um método que requer um int como um argumento de entrada e retorna uma cadeia de
caracteres:
Depois que uma variável é declarada, ela não pode ser declarada novamente com um novo tipo e não pode ter
um valor atribuído que não seja compatível com seu tipo declarado. Por exemplo, você não pode declarar um int
e, em seguida, atribuir a ele um valor booliano de true . No entanto, os valores podem ser convertidos em
outros tipos, por exemplo, quando são passados como argumentos de método ou atribuídos a novas variáveis.
Uma conversão de tipo que não causa a perda de dados é executada automaticamente pelo compilador. Uma
conversão que pode causar perda de dados requer um cast no código-fonte.
Para obter mais informações, consulte Conversões e conversões de Tipo.
Tipos internos
O C# fornece um conjunto padrão de tipos numéricos internos para representar números inteiros, valores de
ponto flutuante, expressões boolianas, caracteres de texto, valores decimais e outros tipos de dados. Também há
tipos string e object internos. Eles estão disponíveis para uso em qualquer programa em C#. Para obter mais
informações sobre os tipos internos, consulte Tabelas de referência de tipos internos.
Tipos personalizados
Você usa os constructos struct, classe, interface e enum para criar seus próprios tipos personalizados. A
biblioteca de classes do .NET em si é uma coleção de tipos personalizados fornecida pela Microsoft, que você
pode usar em seus próprios aplicativos. Por padrão, os tipos usados com mais frequência na biblioteca de
classes estão disponíveis em qualquer programa em C#. Outros ficam disponíveis somente quando você
adiciona explicitamente uma referência de projeto ao assembly no qual eles estão definidos. Após o compilador
ter uma referência ao assembly, você pode declarar variáveis (e constantes) dos tipos declarados no assembly no
código-fonte. Para saber mais, confira Biblioteca de classes do .NET.
NOTE
Você pode ver que os tipos mais usados normalmente são todos organizados no namespace System. No entanto, o
namespace no qual um tipo está contido não tem relação com a possibilidade de ele ser um tipo de valor ou um tipo de
referência.
Tipos de valor
Os tipos de valor derivam de System.ValueType, que deriva de System.Object. Os tipos que derivam de
System.ValueType apresentam um comportamento especial no CLR. As variáveis de tipo de valor contêm
diretamente seus valores, o que significa que a memória é alocada embutida em qualquer contexto em que a
variável é declarada. Não há nenhuma alocação de heap separada ou sobrecarga de coleta de lixo para variáveis
do tipo de valor.
Há duas categorias de tipos de valor: struct e enum.
Os tipos numéricos internos são structs e têm propriedades e métodos que você pode acessar:
// Static method on type byte.
byte b = byte.MaxValue;
Mas você declara e atribui valores a eles como se fossem tipos de não agregação simples:
Tipos de valor são lacrados, o que significa, por exemplo, que você não pode derivar um tipo de System.Int32 e
não pode definir um struct para herdar de qualquer struct ou classe definida pelo usuário, porque um struct
apenas pode herdar de System.ValueType. No entanto, um struct pode implementar uma ou mais interfaces. É
possível converter um tipo de struct em qualquer tipo de interface que ele implementa. Isso faz com que uma
operação de conversão boxing envolva o struct dentro de um objeto de tipo de referência no heap gerenciado.
As operações de conversão boxing ocorrem quando você passa um tipo de valor para um método que usa um
System.Object ou qualquer tipo de interface como parâmetro de entrada. Para obter mais informações, consulte
Boxing e unboxing.
Você usa a palavra-chave struct para criar seus próprios tipos de valor personalizados. Normalmente, um struct
é usado como um contêiner para um pequeno conjunto de variáveis relacionadas, conforme mostrado no
exemplo a seguir:
Para obter mais informações sobre structs, consulte Structs. Para saber mais sobre os tipos de valor do .NET,
confira Tipos de valor.
A outra categoria de tipos de valor é enum. Uma enum define um conjunto de constantes integrais nomeadas.
Por exemplo, a enumeração System.IO.FileMode na biblioteca de classes do .NET contém um conjunto de
números inteiros constantes nomeados que especificam como um arquivo deve ser aberto. Ela é definida
conforme mostrado no exemplo abaixo:
A constante System.IO.FileMode.Create tem um valor de 2. No entanto, o nome é muito mais significativo para a
leitura do código-fonte por humanos e, por esse motivo, é melhor usar enumerações em vez de números literais
constantes. Para obter mais informações, consulte System.IO.FileMode.
Todas as enumerações herdam de System.Enum, que herda de System.ValueType. Todas as regras que se
aplicam a structs também se aplicam a enums. Para obter mais informações sobre enums, consulte Tipos de
enumeração.
Tipos de referência
Um tipo que é definido como uma classe, delegado, matriz ou interface é um tipo de referência. No tempo de
execução, quando você declara uma variável de um tipo de referência, a variável contém o valor null até você
criar explicitamente um objeto usando o operador new ou atribuir a ele um objeto que foi criado em outro lugar
com o operador new , conforme mostrado no exemplo a seguir:
Uma interface deve ser inicializada com um objeto da classe que a implementa. Se MyClass implementa
IMyInterface , você cria uma instância de IMyInterface , conforme mostrado no exemplo a seguir:
Quando o objeto é criado, a memória é alocada no heap gerenciado e a variável contém apenas uma referência
para o local do objeto. Os tipos no heap gerenciado requerem sobrecarga quando são alocados e quando são
recuperados pela funcionalidade de gerenciamento automático de memória do CLR, que é conhecida como
coleta de lixo. No entanto, a coleta de lixo também é altamente otimizada e na maioria dos cenários não cria um
problema de desempenho. Para obter mais informações sobre a coleta de lixo, consulte Gerenciamento
automático de memória.
Todas as matrizes são tipos de referência, mesmo se seus elementos forem tipos de valor. As matrizes derivam
implicitamente da classe System.Array, mas você declara e usa as matrizes com a sintaxe simplificada fornecida
pelo C#, conforme mostrado no exemplo a seguir:
Os tipos de referência dão suporte completo à herança. Ao criar uma classe, você pode herdar de outra interface
ou classe que não está definida como lacrada, e outras classes podem herdar de sua classe e substituir os
métodos virtuais. Para obter mais informações sobre como criar suas próprias classes, consulte Classes e
structs. Para obter mais informações sobre herança e métodos virtuais, consulte Herança.
Tipos genéricos
Um tipo pode ser declarado com um ou mais parâmetros de tipo que servem como um espaço reservado para
o tipo real (o tipo concreto) que o código do cliente fornecerá ao criar uma instância do tipo. Esses tipos são
chamados de tipos genéricos. Por exemplo, o tipo .NET System.Collections.Generic.List<T> tem um parâmetro
de tipo que, por convenção, recebe o nome t. Ao criar uma instância do tipo, você especifica o tipo dos objetos
que a lista conterá, por exemplo, Cadeia de caracteres:
O uso do parâmetro de tipo possibilita a reutilização da mesma classe para conter qualquer tipo de elemento
sem precisar converter cada elemento em objeto. As classes de coleção genéricas são chamadas de coleções
fortemente tipadas porque o compilador sabe o tipo específico dos elementos da coleção e pode gerar um erro
em tempo de compilação se, por exemplo, você tentar adicionar um inteiro ao objeto stringList no exemplo
anterior. Para obter mais informações, consulte Genéricos.
Seções relacionadas
Para mais informações, consulte os seguintes tópicos:
Transmissões e conversões de tipo
Conversão boxing e unboxing
Usando o tipo dynamic
Tipos de valor
Tipos de referência
Classes e Structs
Tipos Anônimos
Genéricos
Especificação da linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a fonte
definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
Conversão de tipos de dados XML
Tipos integrais
Coerções e conversões de tipo (Guia de
Programação em C#)
25/11/2019 • 9 minutes to read • Edit Online
Como o C# é tipado estaticamente no tempo de compilação, depois que uma variável é declarada, ela não pode
ser declarada novamente ou atribuída a um valor de outro tipo, a menos que esse tipo possa ser convertido
implicitamente no tipo da variável. Por exemplo, a string não pode ser convertida implicitamente em int .
Portanto, depois de declarar i como um int , não é possível atribuir a cadeia de caracteres "Hello" a ele,
como mostra o código a seguir:
int i;
i = "Hello"; // error CS0029: Cannot implicitly convert type 'string' to 'int'
No entanto, às vezes é necessário copiar um valor para uma variável ou um parâmetro de método de outro
tipo. Por exemplo, você pode ter que passar uma variável de inteiro para um método cujo parâmetro é digitado
como double . Ou talvez precise atribuir uma variável de classe a uma variável de um tipo de interface. Esses
tipos de operações são chamados de conversões de tipo. No C#, você pode realizar os seguintes tipos de
conversões:
Conversões implícitas: nenhuma sintaxe especial é necessária porque a conversão é de tipo seguro e
nenhum dado será perdido. Exemplos incluem conversões de tipos integrais menores para maiores e
conversões de classes derivadas para classes base.
Conversões explícitas (conversão) : conversões explícitas exigem o operador cast () . A conversão é
necessária quando as informações podem ser perdidas na conversão ou quando a conversão pode não
funcionar por outros motivos. Exemplos típicos incluem a conversão numérica para um tipo que tem
menos precisão ou um intervalo menor e a conversão de uma instância de classe base para uma classe
derivada.
Conversões definidas pelo usuário: as conversões definidas pelo usuário são realizadas por métodos
especiais que podem ser definidos para habilitar conversões explícitas e implícitas entre tipos
personalizados que não têm uma relação de classe base/classe derivada. Para saber mais, confira
Operadores de conversão definidos pelo usuário.
Conversões com classes auxiliares: para converter entre tipos não compatíveis, assim como inteiros e
objetos System.DateTime, ou cadeias de caracteres hexadecimais e matrizes de bytes, você pode usar a
classe System.BitConverter, a classe System.Convert e os métodos Parse dos tipos numéricos internos,
tais como Int32.Parse. Para obter mais informações, consulte Como converter uma matriz de bytes em
um int, Como converter uma cadeia de caracteres em um número e Como converter entre cadeias de
caracteres hexadecimais e tipos numéricos.
Conversões implícitas
Para tipos numéricos internos, uma conversão implícita poderá ser feita quando o valor a ser armazenado
puder se ajustar à variável sem ser truncado ou arredondado. Para tipos integrais, isso significa que o intervalo
do tipo de origem é um subconjunto apropriado do intervalo para o tipo de destino. Por exemplo, uma variável
do tipo long (inteiro de 64 bits) pode armazenar qualquer valor que um int (inteiro de 32 bits) pode armazenar.
No exemplo a seguir, o compilador converte implicitamente o valor de num à direita em um tipo long antes
de atribuí-lo a bigNum .
// Implicit conversion. A long can
// hold any value an int can hold, and more!
int num = 2147483647;
long bigNum = num;
Para obter uma lista completa de todas as conversões numéricas implícitas, consulte a seção conversões
numéricas implícitas do artigo sobre conversões numéricas internas .
Para tipos de referência, uma conversão implícita sempre existe de uma classe para qualquer uma das suas
interfaces ou classes base diretas ou indiretas. Nenhuma sintaxe especial é necessária porque uma classe
derivada sempre contém todos os membros de uma classe base.
Conversões explícitas
No entanto, se uma conversão não puder ser realizada sem o risco de perda de informações, o compilador
exigirá que você execute uma conversão explícita, que é chamada de cast. Uma conversão é uma maneira de
informar explicitamente ao compilador que você pretende fazer a conversão e que você está ciente de que
poderá ocorrer perda de dados. Para executar uma conversão, especifique entre parênteses o tipo para o qual
você está convertendo, na frente do valor ou da variável a ser convertida. O programa a seguir converte um
Double em um int. O programa não será compilado sem a conversão.
class Test
{
static void Main()
{
double x = 1234.7;
int a;
// Cast double to int.
a = (int)x;
System.Console.WriteLine(a);
}
}
// Output: 1234
Para obter uma lista completa de conversões numéricas explícitas com suporte, consulte a seção de conversões
numéricas explícitas do artigo embutir conversões numéricas .
Para tipos de referência, uma conversão explícita será necessária se você precisar converter de um tipo base
para um tipo derivado:
Uma operação de conversão entre tipos de referência não altera o tipo de tempo de execução do objeto
subjacente. Ela apenas altera o tipo do valor que está sendo usado como uma referência a esse objeto. Para
obter mais informações, consulte Polimorfismo.
using System;
class Animal
{
public void Eat() { Console.WriteLine("Eating."); }
public override string ToString()
{
return "I am an animal.";
}
}
class Reptile : Animal { }
class Mammal : Animal { }
class UnSafeCast
{
static void Main()
{
Test(new Mammal());
O C# fornece o operador is para habilitar o teste de compatibilidade antes de realmente executar uma
conversão. Para obter mais informações, consulte como converter com segurança usando correspondência de
padrões e os operadores as e is.
Especificação da linguagem C#
Para saber mais, confira a seção Conversões da Especificação da linguagem C#.
Consulte também
Guia de Programação em C#
Tipos
Operador cast ()
Operadores de conversão definidos pelo usuário
Conversão de tipos generalizados
Como converter uma cadeia de caracteres em um número
Conversões boxing e unboxing (Guia de
Programação em C#)
23/10/2019 • 9 minutes to read • Edit Online
Conversão boxing é o processo de conversão de um tipo de valor para o tipo object ou para qualquer tipo de
interface implementada por esse tipo de valor. Quando o CLR realiza a conversão boxing de um tipo de valor, ele
encapsula o valor dentro de uma instância do System.Object e o armazena no heap gerenciado. A conversão
unboxing extrai o tipo de valor do objeto. A conversão boxing é implícita, a conversão unboxing é explícita. O
conceito de conversões boxing e unboxing serve como base para a exibição unificada de C# do sistema de tipos
em que um valor de qualquer tipo pode ser tratado como um objeto.
No exemplo a seguir, a variável de inteiro i é submetida à conversão boxing e atribuída ao objeto o .
int i = 123;
// The following line boxes i.
object o = i;
o = 123;
i = (int)o; // unboxing
// String.Concat example.
// String.Concat has many versions. Rest the mouse pointer on
// Concat in the following statement to verify that the version
// that is used here takes three object arguments. Both 42 and
// true must be boxed.
Console.WriteLine(String.Concat("Answer", 42, true));
// List example.
// Create a list of objects to hold a heterogeneous collection
// of elements.
List<object> mixedList = new List<object>();
// The following loop sums the squares of the first group of boxed
// integers in mixedList. The list elements are objects, and cannot
// be multiplied or added to the sum until they are unboxed. The
// unboxing must be done explicitly.
var sum = 0;
for (var j = 1; j < 5; j++)
{
// The following statement causes a compiler error: Operator
// '*' cannot be applied to operands of type 'object' and
// 'object'.
//sum += mixedList[j] * mixedList[j]);
// Output:
// Answer42True
// First Group:
// 1
// 2
// 3
// 4
// Second Group:
// 5
// 6
// 7
// 8
// 9
// Sum: 30
Desempenho
Em relação às atribuições simples, as conversões boxing e unboxing são processos computacionalmente
dispendiosos. Quando um tipo de valor é submetido à conversão boxing, um novo objeto deve ser alocado e
construído. A um grau menor, a conversão necessária para a conversão unboxing também é
computacionalmente dispendiosa. Para obter mais informações, consulte Desempenho.
Boxing
A conversão boxing é usada para armazenar tipos de valor no heap coletado como lixo. A conversão boxing é
uma conversão implícita de um tipo de valor para o tipo object ou para qualquer tipo de interface
implementada por esse tipo de valor. A conversão boxing de um tipo de valor aloca uma instância de objeto no
heap e copia o valor no novo objeto.
Considere a seguinte declaração de uma variável de tipo de valor:
int i = 123;
A instrução a seguir aplica implicitamente a operação de conversão boxing na variável i :
O resultado dessa instrução é a criação de uma referência de objeto o , na pilha, que faz referência a um valor
do tipo int , no heap. Esse valor é uma cópia do valor do tipo de valor atribuído à variável i . A diferença entre
as duas variáveis, i e o , é ilustrada na figura de conversão boxing a seguir:
Também é possível executar a conversão boxing explicitamente como no exemplo a seguir, mas a conversão
boxing explícita nunca é necessária:
int i = 123;
object o = (object)i; // explicit boxing
DESCRIÇÃO
Este exemplo converte uma variável de inteiro i em um objeto o usando a conversão boxing. Em seguida, o
valor armazenado na variável i é alterado de 123 para 456 . O exemplo mostra que o tipo do valor original e
o objeto submetido à conversão boxing usa locais de memória separados e, portanto, pode armazenar valores
diferentes.
Exemplo
class TestBoxing
{
static void Main()
{
int i = 123;
Unboxing
A conversão unboxing é uma conversão explícita do tipo object para um tipo de valor ou de um tipo de
interface para um tipo de valor que implementa a interface. Uma operação de conversão unboxing consiste em:
Verificar a instância do objeto para garantir que ele é um valor da conversão boxing de um determinado
tipo de valor.
Copiar o valor da instância para a variável de tipo de valor.
As instruções a seguir demonstram operações conversão boxing e unboxing:
Para a conversão unboxing de tipos de valor ter êxito em tempo de execução, o item sendo submetido à
conversão unboxing deve ser uma referência para um objeto que foi criado anteriormente ao realizar a
conversão boxing de uma instância desse tipo de valor. Tentar realizar a conversão unboxing de null causa uma
NullReferenceException. Tentar realizar a conversão unboxing de uma referência para um tipo de valor
incompatível causa uma InvalidCastException.
Exemplo
O exemplo a seguir demonstra um caso de conversão unboxing inválida e o InvalidCastException resultante.
Usando try e catch , uma mensagem de erro é exibida quando o erro ocorre.
class TestUnboxing
{
static void Main()
{
int i = 123;
object o = i; // implicit boxing
try
{
int j = (short)o; // attempt to unbox
System.Console.WriteLine("Unboxing OK.");
}
catch (System.InvalidCastException e)
{
System.Console.WriteLine("{0} Error: Incorrect unboxing.", e.Message);
}
}
}
int j = (short) o;
para:
int j = (int) o;
Especificação da Linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a fonte
definitiva para a sintaxe e o uso de C#.
Seções relacionadas
Para saber mais:
Tipos de referência
Tipos de valor
Consulte também
Guia de Programação em C#
Como converter uma matriz de bytes em um int
(Guia de Programação em C#)
31/10/2019 • 2 minutes to read • Edit Online
Este exemplo mostra como usar a classe BitConverter para converter uma matriz de bytes em um int e de volta
em uma matriz de bytes. Talvez você precise converter bytes em um tipo de dados interno depois de ler bytes da
rede, por exemplo. Além do método ToInt32 (Byte[], Int32) no exemplo, a tabela a seguir lista os métodos na
classe BitConverter que convertem bytes (de uma matriz de bytes) em outros tipos internos.
Exemplo
Este exemplo Inicializa uma matriz de bytes, inverte a matriz se a arquitetura do computador for little-endian (ou
seja, o byte menos significativo é armazenado primeiro) e, em seguida, chama o método ToInt32 (byte[], Int32)
para converter quatro bytes na matriz para um int . O segundo argumento para ToInt32 (Byte[], Int32) especifica
o índice de início da matriz de bytes.
NOTE
A saída pode diferir dependendo da ordenação da arquitetura do seu computador.
byte[] bytes = { 0, 0, 0, 25 };
Exemplo
Neste exemplo, o método GetBytes(Int32) da classe BitConverter é chamado para converter um int em uma
matriz de bytes.
NOTE
A saída pode diferir dependendo da ordenação da arquitetura do seu computador.
Consulte também
BitConverter
IsLittleEndian
Tipos
Como converter uma cadeia de caracteres em um
número (Guia de Programação em C#)
29/11/2019 • 7 minutes to read • Edit Online
É possível converter uma cadeia de caracteres em um número chamando o método Parse ou TryParse
encontrado nos diversos tipos numéricos ( int , long , double etc.) ou usando os métodos na classe
System.Convert.
Caso haja uma cadeia de caracteres, será um pouco mais eficiente e simples chamar um método TryParse (por
exemplo, int.TryParse("11", out number) ) ou o método Parse (por exemplo, var number = int.Parse("11") ).
Usar um método Convert é mais útil para objetos gerais que implementam IConvertible.
É possível usar métodos Parse ou TryParse no tipo numérico que se espera que a cadeia de caracteres
contenha, como o tipo System.Int32. O método Convert.ToInt32 usa Parse internamente. O método Parse retorna
o número convertido; o método TryParse retorna um valor Boolean que indica se a conversão foi bem-sucedida
e retorna o número convertido em um parâmetro out . Se a cadeia de caracteres não estiver em um formato
válido, Parse lançará uma exceção, enquanto TryParse retornará false . Ao chamar um método Parse , você
sempre deve usar o tratamento de exceções para capturar um FormatException no caso da operação de análise
falhar.
try
{
int numVal = Int32.Parse("-105");
Console.WriteLine(numVal);
}
catch (FormatException e)
{
Console.WriteLine(e.Message);
}
// Output: -105
try
{
int m = Int32.Parse("abc");
}
catch (FormatException e)
{
Console.WriteLine(e.Message);
}
// Output: Input string was not in a correct format.
O exemplo a seguir ilustra uma abordagem para analisar uma cadeia de caracteres que deve incluir caracteres
numéricos à esquerda (incluindo caracteres hexadecimais) e caracteres não numéricos à direita. Ele atribui
caracteres válidos do início de uma cadeia de caracteres até uma nova cadeia de caracteres antes de chamar o
método TryParse. Como as cadeias de caracteres a ser analisadas contêm um pequeno número de caracteres, o
exemplo chama o método String.Concat para atribuir os caracteres válidos em uma nova cadeia de caracteres.
Para cadeias de caracteres maiores, pode ser usada a classe StringBuilder.
using System;
decimal ToDecimal(String)
float ToSingle(String)
double ToDouble(String)
short ToInt16(String)
int ToInt32(String)
long ToInt64(String)
TIPO NUMÉRICO MÉTODO
ushort ToUInt16(String)
uint ToUInt32(String)
ulong ToUInt64(String)
O exemplo a seguir chama o método Convert.ToInt32(String) para converter uma cadeia de caracteres de entrada
em um int. O exemplo captura as duas exceções mais comuns que podem ser geradas por esse método,
FormatException e OverflowException. Se o número resultante puder ser incrementado sem exceder
Int32.MaxValue, o exemplo adicionará 1 ao resultado e exibirá a saída.
using System;
while (repeat)
{
Console.Write("Enter a number between −2,147,483,648 and +2,147,483,647 (inclusive): ");
Consulte também
Tipos
Como determinar se uma cadeia de caracteres representa um valor numérico
Amostra: Utilitário de Formatação do WinForms do .NET Core (C#)
Como converter entre cadeias de caracteres
hexadecimais e tipos numéricos (Guia de
Programação em C#)
27/11/2019 • 4 minutes to read • Edit Online
Exemplo
Este exemplo gera o valor hexadecimal de cada caractere em um string . Primeiro, ele analisa o string como
uma matriz de caracteres. Em seguida, ele chama ToInt32(Char) em cada caractere para obter seu valor numérico.
Por fim, ele formata o número como sua representação hexadecimal em um string .
Exemplo
Este exemplo analisa um string de valores hexadecimais e gera o caractere correspondente a cada valor
hexadecimal. Primeiro, ele chama o método Split(Char[]) para obter cada valor hexadecimal como um string
individual em uma matriz. Em seguida, ele chama ToInt32(String, Int32) para converter o valor hexadecimal em
um valor decimal representado como um int. Ele mostra duas maneiras diferentes de obter o caractere
correspondente a esse código de caractere. A primeira técnica usa ConvertFromUtf32(Int32), que retorna o
caractere correspondente ao argumento de inteiro como um string . A segunda técnica converte explicitamente
o int em um char.
Exemplo
Este exemplo mostra outra maneira de converter um hexadecimal string em um inteiro, chamando o método
Parse(String, NumberStyles).
Exemplo
O exemplo a seguir mostra como converter um hexadecimal string para um float usando a classe
System.BitConverter e o método UInt32.Parse.
// Output: 200.0056
Exemplo
O exemplo a seguir mostra como converter uma matriz de bytes em uma cadeia de caracteres hexadecimal
usando a classe System.BitConverter.
/*Output:
01-AA-B1-DC-10-DD
01AAB1DC10DD
*/
Consulte também
Cadeias de Caracteres de Formato Numérico Padrão
Tipos
Como determinar se uma cadeia de caracteres representa um valor numérico
Usando o tipo dynamic (Guia de Programação em
C#)
04/11/2019 • 9 minutes to read • Edit Online
O C# 4 apresenta um novo tipo, dynamic . O tipo é um tipo estático, mas um objeto do tipo dynamic ignora a
verificação de tipo estático. Na maioria dos casos, ele funciona como se tivesse o tipo object . Em tempo de
compilação, supõem-se que um elemento que tem o tipo dynamic dá suporte a qualquer operação. Portanto,
você não precisa se preocupar se o objeto obtém seu valor de uma API COM, de uma linguagem dinâmica como
o IronPython, do HTML DOM (Modelo de Objeto do Documento), a reflexão ou de algum outro lugar no
programa. No entanto, se o código não for válido, os erros serão capturados em tempo de execução.
Por exemplo, se método de instância exampleMethod1 no código a seguir tiver apenas um parâmetro, o
compilador reconhecerá que a primeira chamada para o método, ec.exampleMethod1(10, 4) , não é válido porque
ele contém dois argumentos. Essa chamada causa um erro do compilador. A segunda chamada para o método,
dynamic_ec.exampleMethod1(10, 4) , não é verificada pelo compilador porque o tipo de dynamic_ec é dynamic .
Portanto, nenhum erro de compilador é relatado. No entanto, o erro não escapa o aviso indefinidamente. Ele é
detectado em tempo de execução e causa uma exceção de tempo de execução.
class ExampleClass
{
public ExampleClass() { }
public ExampleClass(int v) { }
A função do compilador nesses exemplos é reunir informações sobre o que cada instrução está propondo fazer
para o objeto ou expressão que tem o tipo dynamic . Em tempo de execução, as informações armazenadas são
examinadas e qualquer instrução inválida causa uma exceção de tempo de execução.
O resultado de operações mais dinâmicas é dynamic . Por exemplo, se você passar o ponteiro do mouse sobre o
uso de testSum no exemplo a seguir, o IntelliSense exibirá o tipo (variável local) testSum dinâmico.
dynamic d = 1;
var testSum = d + 3;
// Rest the mouse pointer over testSum in the following statement.
System.Console.WriteLine(testSum);
Conversões
As conversões entre objetos dinâmicos e outros tipos são fáceis. Isso permite que o desenvolvedor mude entre o
comportamento dinâmico e o não dinâmico.
Qualquer objeto pode ser convertido no tipo dinâmico implicitamente, conforme mostrado nos exemplos a
seguir.
dynamic d1 = 7;
dynamic d2 = "a string";
dynamic d3 = System.DateTime.Today;
dynamic d4 = System.Diagnostics.Process.GetProcesses();
Por outro lado, uma conversão implícita pode ser aplicada dinamicamente a qualquer expressão do tipo dynamic .
int i = d1;
string str = d2;
DateTime dt = d3;
System.Diagnostics.Process[] procs = d4;
// Valid.
ec.exampleMethod2("a string");
// The following statement does not cause a compiler error, even though ec is not
// dynamic. A run-time exception is raised because the run-time type of d1 is int.
ec.exampleMethod2(d1);
// The following statement does cause a compiler error.
//ec.exampleMethod2(7);
Tempo de execução de linguagem dinâmico
O DLR (Dynamic Language Runtime) é uma nova API no .NET Framework 4. Ele fornece a infraestrutura que dá
suporte ao tipo dynamic em C# e também à implementação das linguagens de programação dinâmicas como
IronPython e IronRuby. Para obter mais informações sobre o DLR, consulte Visão geral do Dynamic Language
Runtime.
interoperabilidade COM
O C# 4 inclui vários recursos que aprimoram a experiência de interoperar com APIs COM, como as APIs de
Automação do Office. Entre os aperfeiçoamentos estão o uso do tipo dynamic e de argumentos nomeados e
opcionais.
Muitos métodos COM permitem variação nos tipos de argumento e tipo de retorno, especificando os tipos como
object . Isso exigiu a conversão explícita dos valores para coordenar com variáveis fortemente tipadas no C#. Se
você compilar usando a opção -link (C# opções do compilador) , a introdução do tipo de dynamic permite tratar
as ocorrências de object em assinaturas com como se fossem do tipo dynamic e, portanto, evitar grande parte
da conversão. Por exemplo, as seguintes instruções de contrastam como acessar uma célula em uma planilha do
Microsoft Office Excel com o tipo dynamic e sem o tipo dynamic .
// After the introduction of dynamic, the access to the Value property and
// the conversion to Excel.Range are handled by the run-time COM binder.
excelApp.Cells[1, 1].Value = "Name";
Excel.Range range2010 = excelApp.Cells[1, 1];
Tópicos relacionados
TÍTULO DESCRIÇÃO
Visão geral do Dynamic Language Runtime Fornece uma visão geral do DLR, que é um ambiente de
tempo de execução que adiciona um conjunto de serviços
para as linguagens dinâmicas para o CLR (Common
Language Runtime).
Passo a passo: Criando e usando objetos dinâmicos Fornece instruções passo a passo para criar um objeto
dinâmico personalizado e para criar um projeto que acessa
uma biblioteca IronPython .
Como acessar objetos de interoperabilidade do Office usando Demonstra como criar um projeto que usa argumentos
recursos do Visual C# nomeados e opcionais, o tipo dynamic e outros
aprimoramentos que simplificam o acesso aos objetos de API
do Office.
Passo a passo: Criando e usando objetos dinâmicos
(C# e Visual Basic)
04/11/2019 • 19 minutes to read • Edit Online
Os objetos dinâmicos expõem membros como propriedades e métodos em tempo de execução, em vez de em
tempo de compilação. Isso permite que você crie objetos para trabalhar com estruturas que não correspondem a
um formato ou tipo estático. Por exemplo, você pode usar um objeto dinâmico para fazer referência ao DOM
(Modelo de Objeto do Documento) HTML, que pode conter qualquer combinação de atributos e elementos de
marcação HTML válidos. Como cada documento HTML é único, os membros de um determinado documento
HTML são determinados em tempo de execução. Um método comum para fazer referência a um atributo de um
elemento HTML é passar o nome do atributo para o método GetProperty do elemento. Para fazer referência ao
atributo id do elemento HTML <div id="Div1"> , primeiro você obtém uma referência ao elemento <div> e,
depois, usa divElement.GetProperty("id") . Se usar um objeto dinâmico, você poderá fazer referência ao atributo
id como divElement.id .
Objetos dinâmicos também fornecem acesso conveniente a linguagens dinâmicas, como IronPython e IronRuby. É
possível usar um objeto dinâmico para fazer referência a um script dinâmico que é interpretado em tempo de
execução.
Você faz referência a um objeto dinâmico usando a associação tardia. Em C#, você especifica o tipo de um objeto
com associação tardia como dynamic . No Visual Basic, você especifica o tipo de um objeto de associação tardia
como Object . Para obter mais informações, consulte dynamic e Associação antecipada e tardia.
Você pode criar objetos dinâmicos personalizados usando classes no namespace System.Dynamic. Por exemplo, é
possível criar um ExpandoObject e especificar os membros desse objeto em tempo de execução. Você também
pode criar seu próprio tipo que herda da classe DynamicObject. Em seguida, você pode substituir os membros da
classe DynamicObject para fornecer funcionalidade dinâmica de tempo de execução.
Nesta explicação passo a passo, você executará as seguintes tarefas:
Criar um objeto personalizado que expõe dinamicamente o conteúdo de um arquivo de texto como
propriedades de um objeto.
Criar um projeto que usa uma biblioteca IronPython .
Prerequisites
É necessário ter o IronPython para .NET para concluir este passo a passo. Navegue até a respectiva página de
download para obter a versão mais recente.
NOTE
Seu computador pode mostrar diferentes nomes ou locais para alguns dos elementos de interface do usuário do Visual
Studio nas instruções a seguir. A edição do Visual Studio que você possui e as configurações que você usa determinam esses
elementos. Para obter mais informações, consulte Personalizando o IDE.
using System.IO;
using System.Dynamic;
Imports System.IO
Imports System.Dynamic
6. O objeto dinâmico personalizado usa um enum para determinar os critérios de pesquisa. Antes da
instrução de classe, adicione a seguinte definição de enum.
7. Atualize a instrução de classe para herdar a classe DynamicObject , conforme mostrado no exemplo de
código a seguir.
class ReadOnlyFile : DynamicObject
8. Adicione o código a seguir para a classe ReadOnlyFile para definir um campo particular para o caminho do
arquivo e um construtor para a classe ReadOnlyFile .
// Store the path to the file and the initial line count value.
private string p_filePath;
// Public constructor. Verify that file exists and store the path in
// the private variable.
public ReadOnlyFile(string filePath)
{
if (!File.Exists(filePath))
{
throw new Exception("File path does not exist.");
}
p_filePath = filePath;
}
' Store the path to the file and the initial line count value.
Private p_filePath As String
' Public constructor. Verify that file exists and store the path in
' the private variable.
Public Sub New(ByVal filePath As String)
If Not File.Exists(filePath) Then
Throw New Exception("File path does not exist.")
End If
p_filePath = filePath
End Sub
try
{
sr = new StreamReader(p_filePath);
while (!sr.EndOfStream)
{
line = sr.ReadLine();
switch (StringSearchOption)
{
case StringSearchOption.StartsWith:
if (testLine.StartsWith(propertyName.ToUpper())) { results.Add(line); }
break;
case StringSearchOption.Contains:
if (testLine.Contains(propertyName.ToUpper())) { results.Add(line); }
break;
case StringSearchOption.EndsWith:
if (testLine.EndsWith(propertyName.ToUpper())) { results.Add(line); }
break;
}
}
}
catch
{
// Trap any exception that occurs in reading the file and return null.
results = null;
}
finally
{
if (sr != null) {sr.Close();}
}
return results;
}
Public Function GetPropertyValue(ByVal propertyName As String,
Optional ByVal StringSearchOption As StringSearchOption =
StringSearchOption.StartsWith,
Optional ByVal trimSpaces As Boolean = True) As List(Of String)
Try
sr = New StreamReader(p_filePath)
Return results
End Function
10. Após o método GetPropertyValue , adicione o seguinte código para substituir o método TryGetMember da
classe DynamicObject. O método TryGetMember é chamado quando um membro de uma classe dinâmica
é solicitado e nenhum argumento é especificado. O argumento binder contém informações sobre o
membro referenciado e o argumento result faz referência ao resultado retornado para o membro
especificado. O método TryGetMember retorna um valor booliano que retorna true se o membro
solicitado existe; caso contrário, ele retorna false .
// Implement the TryGetMember method of the DynamicObject class for dynamic member calls.
public override bool TryGetMember(GetMemberBinder binder,
out object result)
{
result = GetPropertyValue(binder.Name);
return result == null ? false : true;
}
' Implement the TryGetMember method of the DynamicObject class for dynamic member calls.
Public Overrides Function TryGetMember(ByVal binder As GetMemberBinder,
ByRef result As Object) As Boolean
result = GetPropertyValue(binder.Name)
Return If(result Is Nothing, False, True)
End Function
11. Após o método TryGetMember , adicione o seguinte código para substituir o método TryInvokeMember da
classe DynamicObject. O método TryInvokeMember é chamado quando um membro de uma classe
dinâmica é solicitado com argumentos. O argumento binder contém informações sobre o membro
referenciado e o argumento result faz referência ao resultado retornado para o membro especificado. O
argumento args contém uma matriz de argumentos que são passados ao membro. O método
TryInvokeMember retorna um valor booliano que retorna true se o membro solicitado existe; caso
contrário, ele retorna false .
A versão personalizada do método TryInvokeMember espera que o primeiro argumento seja um valor do
enum StringSearchOption que você definiu na etapa anterior. O método TryInvokeMember espera que o
segundo argumento seja um valor booliano. Se um ou os dois argumentos forem valores válidos, eles
serão passados para o método GetPropertyValue para recuperar os resultados.
try
{
if (args.Length > 0) { StringSearchOption = (StringSearchOption)args[0]; }
}
catch
{
throw new ArgumentException("StringSearchOption argument must be a StringSearchOption enum
value.");
}
try
{
if (args.Length > 1) { trimSpaces = (bool)args[1]; }
}
catch
{
throw new ArgumentException("trimSpaces argument must be a Boolean value.");
}
Try
If args.Length > 0 Then StringSearchOption = CType(args(0), StringSearchOption)
Catch
Throw New ArgumentException("StringSearchOption argument must be a StringSearchOption enum
value.")
End Try
Try
If args.Length > 1 Then trimSpaces = CType(args(1), Boolean)
Catch
Throw New ArgumentException("trimSpaces argument must be a Boolean value.")
End Try
3. Se você estiver usando o Visual Basic, clique com o botão direito do mouse no projeto
DynamicIronPythonSample e, em seguida, clique em Propriedades. Clique na guia referências . Clique no
botão Adicionar . Se você estiver usando o Visual C#, no Gerenciador de Soluções, clique com botão
direito do mouse na pasta Referências e, em seguida, clique em Adicionar Referência.
4. Na guia Procurar, navegue até a pasta em que as bibliotecas do IronPython estão instaladas. Por exemplo,
C:\Program Files\IronPython 2.6 for .NET 4.0. Selecione as bibliotecas IronPython.dll,
IronPython.Modules.dll, Microsoft.Scripting.dll e Microsoft.Dynamic.dll. Clique em OK.
5. Se você estiver usando o Visual Basic, edite o arquivo Module1.vb. Se estiver usando o Visual C#, edite o
arquivo Program.cs.
6. Na parte superior do arquivo, adicione o código a seguir para importar os namespaces
Microsoft.Scripting.Hosting e IronPython.Hosting das bibliotecas do IronPython.
using Microsoft.Scripting.Hosting;
using IronPython.Hosting;
Imports Microsoft.Scripting.Hosting
Imports IronPython.Hosting
8. Após o código carregar o módulo random.py, adicione o seguinte código para criar uma matriz de inteiros.
A matriz é passada para o método shuffle do módulo random.py, que classifica aleatoriamente os valores
na matriz.
Consulte também
System.Dynamic
System.Dynamic.DynamicObject
Usando o tipo dynamic
Associação Antecipada e Tardia
dynamic
Implementando interfaces dinâmicas (PDF baixável do Microsoft TechNet)
Classes e structs (Guia de Programação em C#)
23/10/2019 • 12 minutes to read • Edit Online
Classes e structs são duas das construções básicas do Common Type System no .NET Framework. Cada um
é, essencialmente, uma estrutura de dados que encapsula um conjunto de dados e os comportamentos que
são uma unidade lógica. Os dados e os comportamentos são os membros da classe ou struct, e eles incluem
seus métodos, propriedades e eventos, e etc., conforme listado neste tópico.
Uma declaração de classe ou struct é como um plano que é usado para criar instâncias ou objetos em tempo
de execução. Se você definir uma classe ou struct chamado Person , Person será o nome do tipo. Se você
declarar e inicializar um p variável do tipo Person , p será considerado um objeto ou uma instância de
Person . Várias instâncias do mesmo tipo Person podem ser criadas, e cada instância pode ter valores
diferentes em suas propriedades e campos.
Uma classe é um tipo de referência. Quando um objeto da classe é criado, a variável à qual o objeto é
atribuído armazena apenas uma referência na memória. Quando a referência de objeto é atribuída a uma
nova variável, a nova variável refere-se ao objeto original. As alterações feitas por meio de uma variável são
refletidas na outra variável porque ambas se referem aos mesmos dados.
Um struct é um tipo de valor. Quando um struct é criado, a variável à qual o struct está atribuído contém os
dados reais do struct. Quando o struct é atribuído a uma nova variável, ele é copiado. A nova variável e a
variável original, portanto, contêm duas cópias separadas dos mesmos dados. As alterações feitas em uma
cópia não afetam a outra cópia.
Em geral, as classes são usadas para modelar o comportamento mais complexo ou dados que serão
modificados depois que um objeto de classe for criado. Os structs são mais adequados para estruturas de
dados pequenas que contêm principalmente dados que não serão modificados depois que o struct for criado.
Para obter mais informações, consulte Classes, Objetos e Structs.
Exemplo
No exemplo a seguir, CustomClass no namespace ProgrammingGuide tem três membros: um construtor de
instância, uma propriedade chamada Number e um método chamado Multiply . O método Main na classe
Program cria uma instância (objeto) de CustomClass , e o método e a propriedade do objeto são acessados
usando a notação de ponto.
using System;
namespace ProgrammingGuide
{
// Class definition.
public class CustomClass
{
// Class members.
//
// Property.
public int Number { get; set; }
// Method.
public int Multiply(int num)
{
return num * Number;
}
// Instance Constructor.
public CustomClass()
{
Number = 0;
}
}
// Another class definition that contains Main, the program entry point.
class Program
{
static void Main(string[] args)
{
// Create an object of type CustomClass.
CustomClass custClass = new CustomClass();
Encapsulamento
Encapsulamento é chamado, ocasionalmente, de primeiro pilar ou princípio da programação orientada a
objeto. De acordo com o princípio de encapsulamento, uma classe ou struct pode especificar qual membro
será codificado fora da classe ou struct. Os métodos e as variáveis que não serão usados fora da classe ou
assembly poderão ser ocultados para limitar erros de codificação potenciais ou explorações maliciosas.
Para obter mais informações sobre classes, consulte Classes e Objetos.
Membros
Todos os métodos, campos, constantes, propriedades e eventos devem ser declarados em um tipo. Eles são
chamados de membros do tipo. No C#, não existem variáveis globais ou métodos como em algumas das
outras linguagens. Até mesmo um ponto de entrada de um programa, o método Main , deve ser declarado
em uma classe ou struct. A lista a seguir inclui todos os vários tipos de membros que podem ser declarados
em uma classe ou struct.
Campos
Constantes
Propriedades
Métodos
Construtores
Eventos
Finalizadores
Indexadores
Operadores
Tipos aninhados
Acessibilidade
Alguns métodos e propriedades devem ser chamados ou acessado pelo código fora de sua classe ou struct,
também conhecido como código de cliente. Outros métodos e propriedades podem ser usados apenas na
classe ou struct em si. É importante limitar o acessibilidade do código para que somente o código do cliente
desejado possa fazer contato. É possível especificar quão acessível seus tipos e seus membros são para o
código do cliente usando os modificadores de acesso público, protegido, interno, interno protegido, privado e
privado protegido. A acessibilidade padrão é private . Para obter mais informações, consulte Modificadores
de acesso.
Herança
Classes (mas não structs) dão suporte ao conceito de herança. Uma classe que deriva de outra classe (a classe
base) contém automaticamente todos os membros públicos, protegidos e internos da classe base, exceto seus
construtores e finalizadores. Para obter mais informações, consulte Herança e Polimorfismo.
As classes podem ser declaradas como abstratas, o que significa que um ou mais dos seus métodos não têm
nenhuma implementação. Embora as classes abstratas não possam ser instanciadas diretamente, elas servem
como classes base para outras classes que fornecem a implementação ausente. As classes também podem
ser declaradas como lacradas para impedir que outras classes herdem delas. Para obter mais informações,
consulte Classes e membros de classes abstratos e lacrados.
Interfaces
Classes e estruturas podem herdar várias interfaces. Herdar de uma interface significa que o tipo implementa
todos os métodos definidos na interface. Para obter mais informações, consulte Interfaces.
Tipos genéricos
Classes e estruturas podem ser definidas com um ou mais parâmetros de tipo. O código do cliente fornece o
tipo quando ele cria uma instância do tipo. Por exemplo a classe List<T> no namespace
System.Collections.Generic é definida com um parâmetro de tipo. O código do cliente cria uma instância de
um List<string> ou List<int> para especificar o tipo que a lista conterá. Para obter mais informações,
consulte Genéricos.
Tipos estáticos
As classes (mas não structs) podem ser declaradas como estáticas. Uma classe estática pode conter apenas
membros estáticos e não pode ser instanciada com a palavra-chave new. Uma cópia da classe é carregada na
memória quando o programa é carregado e seus membros são acessados pelo nome da classe. Classes e
structs podem conter membros estáticos. Para obter mais informações, consulte Classes estáticas e membros
de classes estáticas.
Tipos aninhados
Uma classe ou struct pode ser aninhado em outra classe ou struct. Para obter mais informações, consulte
Tipos aninhados.
Tipos parciais
Você pode definir parte de uma classe, struct ou método em um arquivo de código e outra parte em um
arquivo de código separado. Para obter mais informações, consulte Classes parciais e métodos.
Inicializadores de objeto
Você pode instanciar e inicializar objetos de classe ou struct e coleções de objetos sem chamar explicitamente
seu construtor. Para obter mais informações, consulte Inicializadores de coleção e objeto.
Tipos anônimos
Em situações nas quais não é conveniente ou necessário criar uma classe nomeada, por exemplo, quando
você estiver preenchendo uma lista com estruturas de dados que você não precisa manter ou passar para
outro método, use tipos anônimos. Para obter mais informações, consulte Tipos anônimos.
Métodos de extensão
Você pode "estender" uma classe sem criar uma classe derivada criando um tipo separado cujos métodos
podem ser chamados como se pertencessem ao tipo original. Para obter mais informações, consulte Métodos
de extensão.
Variáveis Locais Tipadas Implicitamente
Dentro de um método de classe ou struct, você pode usar digitação implícita para instruir o compilador para
determinar o tipo correto no tempo de compilação. Para obter mais informações, consulte Variáveis locais de
tipo implícito.
Especificação da Linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a
fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Guia de Programação em C#
Classes (Guia de Programação em C#)
23/10/2019 • 9 minutes to read • Edit Online
Tipos de referência
Um tipo que é definido como uma classe é um tipo de referência. No tempo de execução, quando você
declara uma variável de um tipo de referência, a variável contém o valor null até que você crie explicitamente
uma instância da classe usando o operador new ou atribua a ela um objeto de um tipo compatível que foi
criado em outro lugar, conforme mostrado no exemplo a seguir:
//Declaring another object of the same type, assigning it the value of the first object.
MyClass mc2 = mc;
Quando o objeto é criado, memória suficiente é alocada no heap gerenciado para o objeto específico, e a
variável contém apenas uma referência para o local do objeto. Os tipos no heap gerenciado requerem
sobrecarga quando são alocados e quando são recuperados pela funcionalidade de gerenciamento
automático de memória do CLR, que é conhecida como coleta de lixo. No entanto, a coleta de lixo também é
altamente otimizada e, na maioria dos cenários, não cria um problema de desempenho. Para obter mais
informações sobre a coleta de lixo, consulte Gerenciamento automático de memória e coleta de lixo.
Declarando Classes
As classes são declaradas usando a palavra-chave class, seguida por um identificador exclusivo, conforme
mostrado no exemplo a seguir:
A palavra-chave class é precedida pelo nível de acesso. Como público é usado nesse caso, qualquer pessoa
pode criar instâncias dessa classe. O nome da classe segue a palavra-chave class . O nome da classe deve ser
um nome do identificador válido em C#. O restante da definição é o corpo da classe, em que o
comportamento e os dados são definidos. Campos, propriedades, métodos e eventos em uma classe são
coletivamente denominados de membros de classe.
Criando objetos
Embora eles sejam usados algumas vezes de maneira intercambiável, uma classe e um objeto são coisas
diferentes. Uma classe define um tipo de objeto, mas não é um objeto em si. Um objeto é uma entidade
concreta com base em uma classe e, às vezes, é conhecido como uma instância de uma classe.
Os objetos podem ser criados usando a palavra-chave new, seguida pelo nome da classe na qual ele se
baseará, dessa maneira:
Customer object1 = new Customer();
Quando uma instância de uma classe é criada, uma referência ao objeto é passada de volta para o
programador. No exemplo anterior, object1 é uma referência a um objeto que é baseado em Customer . Esta
referência refere-se ao novo objeto, mas não contém os dados de objeto. Na verdade, você pode criar uma
referência de objeto sem criar um objeto:
Customer object2;
Não recomendamos a criação de referências de objeto como essa, que não faz referência a um objeto, porque
tentar acessar um objeto por meio de uma referência desse tipo falhará em tempo de execução. Entretanto,
essa referência pode ser feita para se referir a um objeto, criando um novo objeto ou atribuindo-a a um objeto
existente, como abaixo:
Esse código cria duas referências de objeto que fazem referência ao mesmo objeto. Portanto, qualquer
alteração no objeto feita por meio de object3 será refletida no usos posteriores de object4 . Como os
objetos que são baseados em classes são referenciados por referência, as classes são conhecidas como tipos
de referência.
Herança de classe
As classes dão suporte completo à herança, uma característica fundamental da programação orientada a
objetos. Ao criar uma classe, você pode herdar de outra interface ou classe que não está definida como selada,
e outras classes podem herdar de sua classe e substituir seus métodos virtuais.
A herança é realizada usando uma derivação, o que significa que uma classe é declarada usando uma classe
base, da qual ela herda o comportamento e os dados. Uma classe base é especificada ao acrescentar dois-
pontos e o nome de classe base depois do nome de classe derivada, dessa maneira:
Quando uma classe declara uma classe base, ela herda todos os membros da classe base, exceto os
construtores. Para obter mais informações, consulte Herança.
Ao contrário do C++, uma classe no C# só pode herdar diretamente de uma classe base. No entanto, como
uma classe base pode herdar de outra classe, uma classe pode herdar indiretamente várias classes base. Além
disso, uma classe pode implementar diretamente mais de uma interface. Para obter mais informações,
consulte Interfaces.
Uma classe pode ser declarada abstract. Uma classe abstrata contém métodos abstratos que têm uma
definição de assinatura, mas não têm implementação. As classes abstratas não podem ser instanciadas. Elas
só podem ser usadas por meio de classes derivadas que implementam os métodos abstratos. Por outro lado,
uma classe lacrada não permite que outras classes sejam derivadas dela. Para obter mais informações,
consulte Classes e Membros de Classes Abstratos e Lacrados.
As definições de classe podem ser divididas entre arquivos de origem diferentes. Para obter mais
informações, consulte Classes e métodos parciais.
Exemplo
No exemplo a seguir, é definida uma classe pública que contém uma propriedade autoimplementada, um
método e um método especial chamado construtor. Para obter mais informações, consulte os tópicos
Propriedades, Métodos, e Construtores. As instâncias da classe são então instanciadas com a palavra-chave
new .
using System;
Especificação da Linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a
fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Guia de Programação em C#
Programação Orientada a Objeto
Polimorfismo
Nomes de identificadores
Membros
Métodos
Construtores
Finalizadores
Objetos
Objetos (Guia de Programação em C#)
04/11/2019 • 9 minutes to read • Edit Online
Uma definição de classe ou struct é como um esquema que especifica o que o tipo pode fazer. Um objeto é
basicamente um bloco de memória que foi alocado e configurado de acordo com o esquema. Um programa pode
criar vários objetos da mesma classe. Objetos também são chamados de instâncias e podem ser armazenados em
uma variável nomeada ou em uma matriz ou coleção. O código de cliente é o código que usa essas variáveis para
chamar os métodos e acessar as propriedades públicas do objeto. Em uma linguagem orientada a objetos, como
o C#, um programa típico consiste em vários objetos que interagem dinamicamente.
NOTE
Tipos estáticos se comportam de modo diferente do que está descrito aqui. Para obter mais informações, consulte Classes
Estáticas e Membros de Classes Estáticas.
class Program
{
static void Main()
{
Person person1 = new Person("Leopold", 6);
Console.WriteLine("person1 Name = {0} Age = {1}", person1.Name, person1.Age);
}
}
/*
Output:
person1 Name = Leopold Age = 6
person2 Name = Molly Age = 16
person1 Name = Molly Age = 16
*/
Como structs são tipos de valor, uma variável de um objeto de struct mantém uma cópia do objeto inteiro.
Instâncias de structs também podem ser criadas usando o operador new , mas isso não é obrigatório, conforme
mostrado no exemplo a seguir:
public struct Person
{
public string Name;
public int Age;
public Person(string name, int age)
{
Name = name;
Age = age;
}
}
A memória de p1 e p2 é alocada na pilha de thread. Essa memória é recuperada em conjunto com o tipo ou
método em que ela é declarada. Esse é um dos motivos pelos quais os structs são copiados na atribuição. Por
outro lado, a memória alocada a uma instância de classe é recuperada automaticamente (o lixo é coletado) pelo
Common Language Runtime quando todas as referências ao objeto tiveram saído do escopo. Não é possível
destruir de forma determinista um objeto de classe, como é possível no C++. Para obter mais informações sobre
a coleta de lixo no .NET Framework, consulte Coleta de lixo.
NOTE
A alocação e a desalocação de memória no heap gerenciado é altamente otimizada no Common Language Runtime. Na
maioria dos casos, não há uma diferença significativa quanto ao custo do desempenho de alocar uma instância da classe no
heap em vez de alocar uma instância de struct na pilha.
if (p2.Equals(p1))
Console.WriteLine("p2 and p1 have the same values.");
A implementação de System.ValueType de Equals usa reflexão porque ela precisa ser capaz de determinar quais
são os campos em qualquer struct. Ao criar seus próprios structs, substitua o método Equals para fornecer um
algoritmo de igualdade eficiente que é específico ao seu tipo.
Para determinar se os valores dos campos em duas instâncias de classe são iguais, você pode usar o método
Equals ou o Operador ==. No entanto, use-os apenas se a classe os tiver substituído ou sobrecarregado para
fornecer uma definição personalizada do que "igualdade" significa para objetos desse tipo. A classe também
pode implementar a interface IEquatable<T> ou a interface IEqualityComparer<T>. As duas interfaces
fornecem métodos que podem ser usados para testar a igualdade de valores. Ao criar suas próprias classes
que substituem Equals , certifique-se de seguir as diretrizes informadas em Como definir a igualdade de valor
para um tipo e Object.Equals(Object).
Seções relacionadas
Para saber mais:
Classes
Structs
Construtores
Finalizadores
Eventos
Consulte também
Guia de Programação em C#
object
Herança
class
struct
Operador new
Common Type System
Structs (Guia de Programação em C#)
25/11/2019 • 2 minutes to read • Edit Online
Os structs compartilham a maioria da mesma sintaxe que as classes. O nome da struct deve ser um nome do
identificador válido em C#. Os structs são mais limitados do que as classes das seguintes maneiras:
Dentro de uma declaração de struct, os campos não podem ser inicializados, a menos que sejam declarados
como const ou estáticos.
Um struct não pode declarar um construtor sem padrão (um construtor sem parâmetros) ou um finalizador.
Os structs são copiados na atribuição. Quando um struct recebe uma nova variável, todos os dados são
copiados e qualquer modificação na nova cópia não altera os dados da cópia original. É importante se
lembrar disso ao trabalhar com coleções de tipos de valor como Dictionary<string, myStruct> .
Os structs são tipos de valor, diferentemente das classes, que são tipos de referência.
Diferentemente das classes, os structs podem ser instanciados sem usar um operador new .
Os structs podem declarar construtores que têm parâmetros.
Um struct não pode herdar de outra estrutura ou classe e ele não pode ser a base de uma classe. Todos os
structs herdam diretamente do ValueType, que herda do Object.
Um struct pode implementar interfaces.
Um struct não pode ser null e uma variável de struct não pode ser atribuída null a menos que a variável
seja declarada como um tipo de valor anulável.
Consulte também
Guia de Programação em C#
Classes e Structs
Classes
Tipos de valor anuláveis
Nomes de identificadores
Usando structs
Como saber a diferença entre passar uma struct e passar uma referência de classe para um método
Usando structs (C# guia de programação)
27/11/2019 • 5 minutes to read • Edit Online
O tipo struct é adequado para representar objetos leves como Point , Rectangle e Color . Embora seja
conveniente representar um ponto como uma classe com Propriedades Auto-implementadas, um struct pode ser
mais eficiente em alguns cenários. Por exemplo, se você declarar uma matriz de 1000 objetos Point , alocará
memória adicional para referenciar cada objeto, nesse caso, um struct será mais barato. Como o .NET já contém
um objeto chamado Point, o struct neste exemplo é nomeado Coords em vez disso.
É um erro definir um construtor sem parâmetros para uma struct. Também é um erro ao inicializar um campo de
instância em um corpo de struct. Você pode inicializar membros de struct externamente acessíveis somente por
meio de um construtor com parâmetros, do construtor sem parâmetro implícito, de um inicializador de objeto ou
acessando os membros individualmente depois que o struct é declarado. Todos os membros particulares ou, de
outro modo, inacessíveis exigem o uso de construtores exclusivamente.
Quando você cria um objeto de struct usando o operador new, ele é criado e o construtor apropriado é chamado
de acordo com a assinatura do construtor. Diferentemente das classes, os structs podem ser instanciados sem
usar o operador new . Nesse caso, não há nenhuma chamada do construtor, o que torna a alocação mais eficiente.
No entanto, os campos permanecerão não atribuídos e o objeto não poderá ser usado até que todos os campos
sejam inicializados. Isso inclui a incapacidade de obter ou definir valores por meio de propriedades.
Se você criar uma instância de um objeto struct usando o construtor sem parâmetros, todos os membros serão
atribuídos de acordo com seus valores padrão.
Ao gravar um construtor com parâmetros para uma struct, você deve inicializar explicitamente todos os membros;
caso contrário, um ou mais membros permanecerão não atribuídos e a estrutura não poderá ser usada,
produzindo o erro CS0171do compilador.
Não há nenhuma herança para structs como há para classes. Um struct não pode herdar de outra estrutura ou
classe e ele não pode ser a base de uma classe. No entanto, os structs herdam da classe base Object. Um struct
pode implementar interfaces e ele faz isso exatamente como as classes.
Você não pode declarar uma classe usando a palavra-chave struct . No C#, as classes e os structs são
semanticamente diferentes. Um struct é um tipo de valor, enquanto uma classe é um tipo de referência. Para obter
mais informações, consulte tipos de valor e tipos de referência.
A menos que você precise de semântica de tipo de referência, uma pequena classe pode ser tratada com mais
eficiência pelo sistema se você declará-la como um struct em vez disso.
Exemplo 1
Este exemplo demonstra struct inicialização usando os construtores sem parâmetros e com parâmetros.
public struct Coords
{
public int x, y;
// Display results.
Console.Write("Coords 1: ");
Console.WriteLine($"x = {coords1.x}, y = {coords1.y}");
Console.Write("Coords 2: ");
Console.WriteLine($"x = {coords2.x}, y = {coords2.y}");
Exemplo 2
Este exemplo demonstra um recurso que é exclusivo para struct. Ele cria um objeto Coords sem usar o operador
new . Se você substituir a palavra struct pela palavra class , o programa não compilará.
// Initialize.
coords1.x = 10;
coords1.y = 20;
// Display results.
Console.Write("Coords 1: ");
Console.WriteLine($"x = {coords1.x}, y = {coords1.y}");
Consulte também
Guia de Programação em C#
Classes e Structs
Structs
Herança (Guia de Programação em C#)
23/10/2019 • 11 minutes to read • Edit Online
A herança, assim como o encapsulamento e o polimorfismo, é uma das três principais características da
programação orientada ao objeto. A herança permite que você crie novas classes que reutilizam, estendem
e modificam o comportamento definido em outras classes. A classe cujos membros são herdados é
chamada classe base e a classe que herda esses membros é chamada classe derivada. Uma classe derivada
pode ter apenas uma classe base direta. No entanto, a herança é transitiva. Se ClassC for derivada de
ClassB e ClassB for derivada de ClassA, ClassC herdará os membros declarados em ClassA e ClassB.
NOTE
Structs não dão suporte a herança, mas podem implementar interfaces. Para obter mais informações, consulte
Interfaces.
Conceitualmente, uma classe derivada é uma especialização da classe base. Por exemplo, se tiver uma
classe base Animal , você pode ter uma classe derivada chamada Mammal e outra classe derivada chamada
Reptile . Um Mammal é um Animal e um Reptile é um Animal , mas cada classe derivada representa
especializações diferentes da classe base.
Quando você define uma classe para derivar de outra classe, a classe derivada obtém implicitamente todos
os membros da classe base, exceto por seus construtores e finalizadores. A classe derivada pode, assim,
reutilizar o código na classe base sem precisar implementá-lo novamente. Na classe derivada, você pode
adicionar mais membros. Dessa forma, a classe derivada estende a funcionalidade da classe base.
A ilustração a seguir mostra uma classe WorkItem que representa um item de trabalho em um processo
comercial. Como todas as classes, ela deriva de System.Object e herda todos os seus métodos. WorkItem
adiciona cinco membros próprios. Eles incluem um construtor, porque os construtores não são herdados. A
classe ChangeRequest herda de WorkItem e representa um tipo específico de item de trabalho.
ChangeRequest adiciona mais dois membros aos membros que herda de WorkItem e de Object. Ele deve
adicionar seu próprio construtor e também adiciona originalItemID . A propriedade originalItemID
permite que a instância ChangeRequest seja associada ao WorkItem original a que a solicitação de alteração
se aplica.
O exemplo a seguir mostra como as relações entre as classes demonstradas na ilustração anterior são
expressos em C#. O exemplo também mostra como WorkItem substitui o método virtual Object.ToString e
como a classe ChangeRequest herda a implementação de WorkItem do método.
//Properties.
protected int ID { get; set; }
protected string Title { get; set; }
protected string Description { get; set; }
protected TimeSpan jobLength { get; set; }
// Method Update enables you to update the title and job length of an
// existing WorkItem object.
public void Update(string title, TimeSpan joblen)
{
this.Title = title;
this.jobLength = joblen;
}
class Program
{
static void Main()
{
// Create an instance of WorkItem by using the constructor in the
// base class that takes three arguments.
WorkItem item = new WorkItem("Fix Bugs",
"Fix all bugs in my code branch",
new TimeSpan(3, 4, 0, 0));
Interfaces
Uma interface é um tipo de referência semelhante a uma classe base abstrata que consiste somente em
membros abstratos. Quando uma classe implementa uma interface, ela deve fornecer uma implementação
para todos os membros da interface. Uma classe pode implementar várias interfaces, mesmo que ela
possa derivar de apenas uma classe base direta.
Interfaces são usadas para definir recursos específicos para classes que não têm necessariamente uma
relação do tipo “é um". Por exemplo, a interface System.IEquatable<T> pode ser implementada por
qualquer classe ou struct que precise habilitar o código do cliente para determinar se dois objetos do tipo
são equivalentes (como quer que o tipo defina a equivalência). IEquatable<T> não implica o mesmo tipo
de relação "é um" existente entre uma classe base e uma classe derivada (por exemplo, um Mammal é um
Animal ). Para obter mais informações, consulte Interfaces.
Consulte também
Guia de Programação em C#
Classes e Structs
class
struct
Polimorfismo (Guia de Programação em C#)
25/11/2019 • 13 minutes to read • Edit Online
O polimorfismo costuma ser chamado de o terceiro pilar da programação orientada a objetos, depois do
encapsulamento e a herança. O polimorfismo é uma palavra grega que significa "de muitas formas" e tem dois
aspectos distintos:
Em tempo de execução, os objetos de uma classe derivada podem ser tratados como objetos de uma
classe base, em locais como parâmetros de método, coleções e matrizes. Quando isso ocorre, o tipo
declarado do objeto não é mais idêntico ao seu tipo de tempo de execução.
As classes base podem definir e implementar métodos virtuais e as classes derivadas podem substituí-
los, o que significa que elas fornecem sua própria definição e implementação. Em tempo de execução,
quando o código do cliente chama o método, o CLR procura o tipo de tempo de execução do objeto e
invoca a substituição do método virtual. Dessa forma, você pode chamar em seu código-fonte um
método de uma classe base e fazer com que a versão de uma classe derivada do método seja executada.
Os métodos virtuais permitem que você trabalhe com grupos de objetos relacionados de maneira uniforme.
Por exemplo, suponha que você tem um aplicativo de desenho que permite que um usuário crie vários tipos de
formas sobre uma superfície de desenho. Você não sabe em tempo de compilação que tipos específicos de
formas que o usuário criará. No entanto, o aplicativo precisa manter controle de todos os diferentes tipos de
formas que são criados e atualizá-los em resposta às ações do mouse do usuário. Você pode usar o
polimorfismo para resolver esse problema em duas etapas básicas:
1. Crie uma hierarquia de classes em que cada classe de forma específica derive de uma classe base
comum.
2. Use um método virtual para invocar o método adequado em qualquer classe derivada por meio de uma
única chamada para o método da classe base.
Primeiro, crie uma classe base chamada Shape e as classes derivadas como Rectangle , Circle e Triangle .
Atribua à classe Shape um método virtual chamado Draw e substitua-o em cada classe derivada para
desenhar a forma especial que a classe representa. Crie um objeto List<Shape> e adicione um círculo, triângulo
e retângulo para ele. Para atualizar a superfície de desenho, use um loop foreach para iterar na lista e chamar o
método Draw em cada objeto Shape na lista. Mesmo que cada objeto na lista tenha um tipo de declaração de
Shape , é o tipo de tempo de execução (a versão de substituição do método em cada classe derivada) que será
invocado.
using System;
using System.Collections.Generic;
// Virtual method
public virtual void Draw()
{
Console.WriteLine("Performing base class drawing tasks");
}
}
}
class Program
{
static void Main(string[] args)
{
// Polymorphism at work #1: a Rectangle, Triangle and Circle
// can all be used whereever a Shape is expected. No cast is
// required because an implicit conversion exists from a derived
// class to its base class.
var shapes = new List<Shape>
{
new Rectangle(),
new Triangle(),
new Circle()
};
/* Output:
Drawing a rectangle
Performing base class drawing tasks
Drawing a triangle
Performing base class drawing tasks
Drawing a circle
Performing base class drawing tasks
*/
Em C#, cada tipo é polimórfico porque todos os tipos, incluindo tipos definidos pelo usuário, herdam de Object.
Os campos não podem ser virtuais, apenas os métodos, propriedades, eventos e indexadores podem ser
virtuais. Quando uma classe derivada substitui um membro virtual, esse membro é chamado, mesmo quando
uma instância dessa classe está sendo acessada como uma instância da classe base. O código a seguir mostra
um exemplo:
BaseClass A = (BaseClass)B;
A.DoWork(); // Also calls the new method.
Os métodos e propriedades virtuais permitem que classes derivadas estendam uma classe base sem a
necessidade de usar a implementação da classe base de um método. Para obter mais informações, consulte
Controle de versão com as palavras-chave override e new. Uma interface fornece uma outra maneira de definir
um método ou conjunto de métodos cuja implementação é deixada para classes derivadas. Para obter mais
informações, consulte Interfaces.
Ocultando membros de classe base com novos membros
Se você quiser que o seu membro derivado tenha o mesmo nome de um membro de uma classe base, mas não
quiser que ele participe da invocação virtual, será possível usar a palavra-chave new. A palavra-chave new é
colocada antes do tipo de retorno de um membro de classe que está sendo substituído. O código a seguir
mostra um exemplo:
Você ainda pode acessar os membros da classe base ocultos a partir do código do cliente, convertendo a
instância da classe derivada a uma instância da classe base. Por exemplo:
BaseClass A = (BaseClass)B;
A.DoWork(); // Calls the old method.
public class A
{
public virtual void DoWork() { }
}
public class B : A
{
public override void DoWork() { }
}
Uma classe derivada pode interromper a herança virtual, declarando uma substituição como sealed. Isso exige
a colocação da palavra-chave sealed antes da palavra-chave override na declaração de membro de classe. O
código a seguir mostra um exemplo:
public class C : B
{
public sealed override void DoWork() { }
}
No exemplo anterior, o método DoWork não será mais virtual para qualquer classe derivada de C. Ele ainda será
virtual para instâncias de C, mesmo se elas forem convertidas em métodos tipo B ou tipo A. Métodos lacrados
podem ser substituídos por classes derivadas usando a palavra-chave new , como mostra o exemplo a seguir:
public class D : C
{
public new void DoWork() { }
}
Neste caso, se DoWork é chamado em D usando uma variável do tipo D, o novo DoWork é chamado. Se uma
variável do tipo C, B ou A é usada para acessar uma instância de D, uma chamada de DoWork seguirá as regras
de herança virtual, encaminhando as chamadas para a implementação de DoWork na classe C.
Acessando membros virtuais da classe base das classes derivadas
A classe derivada que substituiu um método ou propriedade ainda pode acessar o método ou propriedade na
classe base usando a palavra-chave base . O código a seguir mostra um exemplo:
NOTE
Recomendamos que os membros virtuais usem base para chamar a implementação da classe base do membro em sua
própria implementação. Deixar o comportamento da classe base ocorrer permite que a classe derivada se concentre na
implementação de comportamento específico para a classe derivada. Se a implementação da classe base não é chamado,
cabe à classe derivada tornar seu comportamento compatível com o comportamento da classe base.
Nesta seção
Controle de versão com as palavras-chave override e new
Quando usar as palavras-chave override e new
Como substituir o método ToString
Consulte também
Guia de Programação em C#
Herança
Classes e membros de classes abstract e sealed
Métodos
Eventos
Propriedades
Indexadores
Tipos
Controle de versão com as palavras-chave override
e new (Guia de Programação em C#)
31/10/2019 • 9 minutes to read • Edit Online
A linguagem C# foi projetada para que o controle de versão entre classes derivadas e base em diferentes
bibliotecas possa evoluir e manter a compatibilidade com versões anteriores. Isso significa, por exemplo, que a
introdução de um novo membro em uma classe base com o mesmo nome que um membro em uma classe
derivada tem suporte completo pelo C# e não leva a comportamento inesperado. Isso também significa que
uma classe deve declarar explicitamente se um método destina-se a substituir um método herdado ou se um
método é um novo método que oculta um método herdado de nome semelhante.
No C#, as classes derivadas podem conter métodos com o mesmo nome que os métodos da classe base.
O método de classe base deve ser definido como virtual.
Se o método na classe derivada não for precedido pelas palavras-chave new ou override, o compilador
emitirá um aviso e o método se comportará como se a palavra-chave new estivesse presente.
Se o método na classe derivada for precedido pela palavra-chave new , o método será definido como
sendo independente do método na classe base.
Se o método na classe derivada for precedido pela palavra-chave override , os objetos da classe derivada
chamarão esse método em vez do método da classe base.
O método da classe base pode ser chamado de dentro da classe derivada usando a palavra-chave base .
As palavras-chave override , virtual e new também podem ser aplicadas a propriedades, indexadores e
eventos.
Por padrão, os métodos C# não são virtuais. Se um método for declarado como virtual, qualquer classe que
herdar o método pode implementar sua própria versão. Para tornar um método virtual, o modificador virtual é
usado na declaração de método da classe base. A classe derivada pode, em seguida, substituir o método virtual
base usando a palavra-chave override ou ocultar o método virtual na classe base usando a palavra-chave new .
Se nem a palavra-chave override nem a new for especificada, o compilador emitirá um aviso e o método na
classe derivada ocultará o método na classe base.
Para demonstrar isso na prática, suponha por um momento que a Empresa A tenha criado uma classe chamada
GraphicsClass , que seu programa usa. A seguir está GraphicsClass :
class GraphicsClass
{
public virtual void DrawLine() { }
public virtual void DrawPoint() { }
}
Sua empresa usa essa classe e você a usa para derivar sua própria classe, adicionando um novo método:
class GraphicsClass
{
public virtual void DrawLine() { }
public virtual void DrawPoint() { }
public virtual void DrawRectangle() { }
}
A nova versão de GraphicsClass agora contém um método chamado DrawRectangle . Inicialmente, nada ocorre.
A nova versão ainda é compatível em relação ao binário com a versão antiga. Qualquer software que você
implantou continuará a funcionar, mesmo se a nova classe for instalada nesses sistemas de computador. Todas as
chamadas existentes para o método DrawRectangle continuarão a fazer referência à sua versão, em sua classe
derivada.
No entanto, assim que você recompilar seu aplicativo usando a nova versão do GraphicsClass , você receberá
um aviso do compilador, CS0108. Este aviso informa que você deve considerar como deseja que seu método
DrawRectangle se comporte em seu aplicativo.
Se desejar que seu método substitua o novo método de classe base, use a palavra-chave override :
A palavra-chave override garante que todos os objetos derivados de YourDerivedGraphicsClass usarão a versão
da classe derivada de DrawRectangle . Os objetos derivados de YourDerivedGraphicsClass ainda poderão acessar
a versão da classe base de DrawRectangle usando a palavra-chave base:
base.DrawRectangle();
Se você não quiser que seu método substitua o novo método de classe base, as seguintes considerações se
aplicam. Para evitar confusão entre os dois métodos, você pode renomear seu método. Isso pode ser demorado
e propenso a erros e simplesmente não ser prático em alguns casos. No entanto, se seu projeto for relativamente
pequeno, você poderá usar opções de Refatoração do Visual Studio para renomear o método. Para obter mais
informações, consulte Refatorando classes e tipos (Designer de Classe).
Como alternativa, você pode evitar o aviso usando a palavra-chave new na definição da classe derivada:
Usando a palavra-chave new informa ao compilador que sua definição oculta a definição que está contida na
classe base. Este é o comportamento padrão.
Quando DoWork é chamado em uma instância do Derived , o compilador C# tenta primeiro tornar a chamada
compatível com as versões do DoWork originalmente declarado em Derived . Os métodos de substituição não
são considerados como declarados em uma classe, eles são novas implementações de um método declarado em
uma classe base. Somente se o compilador C# não puder corresponder a chamada de método a um método
original no Derived ele tentará corresponder a chamada a um método com o mesmo nome e parâmetros
compatíveis. Por exemplo:
int val = 5;
Derived d = new Derived();
d.DoWork(val); // Calls DoWork(double).
Como a variável valpode ser convertida para um duplo implicitamente, o compilador C# chama
DoWork(double) em vez de DoWork(int) . Há duas formas de evitar isso. Primeiro, evite declarar novos métodos
com o mesmo nome que os métodos virtuais. Segundo, você pode instruir o compilador C# para chamar o
método virtual fazendo-o pesquisar a lista do método de classe base convertendo a instância do Derived para
Base . Como o método é virtual, a implementação de DoWork(int) em Derived será chamada. Por exemplo:
Para obter mais exemplos de new e override , consulte Quando usar as palavras-chave override e new.
Consulte também
Guia de Programação em C#
Classes e Structs
Métodos
Herança
Quando usar as palavras-chave override e new
(Guia de Programação em C#)
23/10/2019 • 15 minutes to read • Edit Online
No C#, um método em uma classe derivada pode ter o mesmo nome que um método na classe base. É possível
especificar a maneira como os métodos interagem usando as palavras-chave new e override. O modificador
override estende o método virtual da classe base e o modificador new oculta um método de classe base
acessível. A diferença é ilustrada nos exemplos deste tópico.
Em um aplicativo de console, declare as duas classes a seguir, BaseClass e DerivedClass . DerivedClass herda de
BaseClass .
class BaseClass
{
public void Method1()
{
Console.WriteLine("Base - Method1");
}
}
bc.Method1();
dc.Method1();
dc.Method2();
bcdc.Method1();
}
// Output:
// Base - Method1
// Base - Method1
// Derived - Method2
// Base - Method1
}
Em seguida, adicione o seguinte método Method2 a BaseClass . A assinatura desse método corresponde à
assinatura do método Method2 em DerivedClass .
Como BaseClass agora tem um método Method2 , uma segunda instrução de chamada pode ser adicionada para
variáveis de BaseClass bc e bcdc , conforme mostrado no código a seguir.
bc.Method1();
bc.Method2();
dc.Method1();
dc.Method2();
bcdc.Method1();
bcdc.Method2();
Quando você compilar o projeto, verá que a adição do método Method2 gera um aviso BaseClass . O aviso
informa que o método Method2 em DerivedClass oculta o método Method2 em BaseClass . É recomendável
usar a palavra-chave new na definição Method2 se você pretende gerar esse resultado. Como alternativa, seria
possível renomear um dos métodos Method2 para resolver o aviso, mas isso nem sempre é prático.
Antes de adicionar new , execute o programa para ver a saída produzida pelas outras instruções de chamada. Os
seguintes resultados são exibidos.
// Output:
// Base - Method1
// Base - Method2
// Base - Method1
// Derived - Method2
// Base - Method1
// Base - Method2
A palavra-chave new preserva as relações que produzem essa saída, mas suprime o aviso. As variáveis que têm
o tipo BaseClass continuam acessando os membros de BaseClass e a variável que tem o tipo DerivedClass
continua acessando os membros em DerivedClass primeiro e, em seguida, considera os membros herdados de
BaseClass .
Para suprimir o aviso, adicione o modificador new para a definição de Method2 em DerivedClass , conforme
mostrado no código a seguir. O modificador pode ser adicionado antes ou depois de public .
Execute o programa novamente para verificar se a saída não foi alterada. Verifique também se o aviso não é mais
exibido. Usando new , você está declarando que está ciente de que o membro que ele modifica oculta um
membro herdado da classe base. Para obter mais informações sobre a ocultação de nome por meio de herança,
consulte Novo modificador.
Para comparar esse comportamento com os efeitos de usar override , adicione o seguinte método a
DerivedClass . O modificador override pode ser adicionado antes ou depois de public .
Adicione o modificador virtual à definição de Method1 em BaseClass . O modificador virtual pode ser
adicionado antes ou depois de public .
Execute o projeto novamente. Observe principalmente as duas últimas linhas da saída a seguir.
// Output:
// Base - Method1
// Base - Method2
// Derived - Method1
// Derived - Method2
// Derived - Method1
// Base - Method2
O uso do modificador override permite que bcdc acesse o método Method1 definido em DerivedClass .
Normalmente, esse é o comportamento desejado em hierarquias de herança. Você quer objetos com valores
criados da classe derivada para usar os métodos definidos na classe derivada. Obtenha esse comportamento
usando override para estender o método da classe base.
O código a seguir contem o exemplo completo.
using System;
using System.Text;
namespace OverrideAndNew
{
class Program
{
static void Main(string[] args)
{
BaseClass bc = new BaseClass();
DerivedClass dc = new DerivedClass();
BaseClass bcdc = new DerivedClass();
// The following two calls do what you would expect. They call
// the methods that are defined in BaseClass.
bc.Method1();
bc.Method2();
// Output:
// Base - Method1
// Base - Method2
// The following two calls do what you would expect. They call
// the methods that are defined in DerivedClass.
dc.Method1();
dc.Method2();
// Output:
// Derived - Method1
// Derived - Method2
class BaseClass
{
public virtual void Method1()
{
Console.WriteLine("Base - Method1");
}
// Define the base class, Car. The class defines two methods,
// DescribeCar and ShowDetails. DescribeCar calls ShowDetails, and each derived
// class also defines a ShowDetails method. The example tests which version of
// ShowDetails is selected, the base class method or the derived class method.
class Car
{
public void DescribeCar()
{
System.Console.WriteLine("Four wheels and an engine.");
ShowDetails();
}
O exemplo testa qual versão do ShowDetails é chamada. O método a seguir, TestCars1 , declara uma instância
de cada classe e, em seguida, chama DescribeCar em cada instância.
public static void TestCars1()
{
System.Console.WriteLine("\nTestCars1");
System.Console.WriteLine("----------");
// Notice the output from this test case. The new modifier is
// used in the definition of ShowDetails in the ConvertibleCar
// class.
TestCars1 produz a saída a seguir. Observe principalmente os resultados para car2 , que provavelmente não
são o que você espera. O tipo do objeto é ConvertibleCar , mas DescribeCar não acessa a versão de
ShowDetails definida na classe ConvertibleCar , porque esse método é declarado com o modificador new , e não
com o modificador override . Em decorrência disso, um objeto ConvertibleCar exibe a mesma descrição que um
objeto Car . Compare os resultados de car3 , que é um objeto Minivan . Nesse caso, o método ShowDetails
declarado na classe Minivan substitui o método ShowDetails declarado na classe Car e a descrição exibida
descreve uma minivan.
// TestCars1
// ----------
// Four wheels and an engine.
// Standard transportation.
// ----------
// Four wheels and an engine.
// Standard transportation.
// ----------
// Four wheels and an engine.
// Carries seven people.
// ----------
TestCars2 cria uma lista de objetos que têm tipo Car . Os valores dos objetos são instanciados com base nas
classes Car , ConvertibleCar e Minivan . DescribeCar é chamado em cada elemento da lista. O código a seguir
mostra a definição de TestCars2 .
// TestCars2
// ----------
// Four wheels and an engine.
// Standard transportation.
// ----------
// Four wheels and an engine.
// Standard transportation.
// ----------
// Four wheels and an engine.
// Carries seven people.
// ----------
Os métodos TestCars3 e TestCars4 completam o exemplo. Esses métodos chamam ShowDetails diretamente,
primeiro com base nos objetos declarados para ter o tipo ConvertibleCar e Minivan ( TestCars3 ), em seguida,
com base nos objetos declarados para ter o tipo Car ( TestCars4 ). O código a seguir define esses dois métodos.
Os métodos produzem a saída a seguir, que corresponde aos resultados do primeiro exemplo neste tópico.
// TestCars3
// ----------
// A roof that opens up.
// Carries seven people.
// TestCars4
// ----------
// Standard transportation.
// Carries seven people.
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace OverrideAndNew2
namespace OverrideAndNew2
{
class Program
{
static void Main(string[] args)
{
// Declare objects of the derived classes and test which version
// of ShowDetails is run, base or derived.
TestCars1();
// Notice the output from this test case. The new modifier is
// used in the definition of ShowDetails in the ConvertibleCar
// class.
ConvertibleCar car2 = new ConvertibleCar();
car2.DescribeCar();
System.Console.WriteLine("----------");
// Define the base class, Car. The class defines two virtual methods,
// DescribeCar and ShowDetails. DescribeCar calls ShowDetails, and each derived
// class also defines a ShowDetails method. The example tests which version of
// ShowDetails is used, the base class method or the derived class method.
class Car
{
public virtual void DescribeCar()
{
System.Console.WriteLine("Four wheels and an engine.");
ShowDetails();
}
Consulte também
Guia de Programação em C#
Classes e Structs
Controle de versão com as palavras-chave override e new
base
abstract
Como substituir o método ToString (guiaC# de
programação)
25/11/2019 • 2 minutes to read • Edit Online
Cada classe ou struct no C# herda implicitamente a classe Object. Portanto, cada objeto no C# obtém o método
ToString, que retorna uma representação de cadeia de caracteres desse objeto. Por exemplo, todas as variáveis do
tipo int tem um método ToString , que permite retornar seus conteúdos como uma cadeia de caracteres:
int x = 42;
string strx = x.ToString();
Console.WriteLine(strx);
// Output:
// 42
Ao criar uma classe ou struct personalizada, é necessário substituir o método ToString a fim de fornecer
informações sobre o tipo ao código cliente.
Para obter informações sobre como usar cadeias de caracteres de formato e outros tipos de formatação
personalizada com o método ToString , consulte Tipos de Formatação.
IMPORTANT
Ao decidir quais informações devem ser fornecidas por meio desse método, considere se a classe ou struct será utilizado por
código não confiável. Assegure-se de que nenhuma informação que possa ser explorada por código mal-intencionado seja
fornecida.
class Person
{
public string Name { get; set; }
public int Age { get; set; }
Consulte também
IFormattable
Guia de Programação em C#
Classes e Structs
Cadeias de Caracteres
string
override
virtual
Formatando Tipos
Membros (Guia de Programação em C#)
31/10/2019 • 3 minutes to read • Edit Online
Classes e structs têm membros que representam seus dados e comportamento. Os membros de uma classe
incluem todos os membros declarados na classe, juntamente com todos os membros (exceto construtores e
finalizadores) declarados em todas as classes em sua hierarquia de herança. Os membros privados em classes
base são herdados, mas não podem ser acessados de classes derivadas.
A tabela a seguir lista os tipos de membros que uma classe ou struct pode conter:
MEMBRO DESCRIÇÃO
Tipos aninhados Os tipos aninhados são tipos declarados dentro de outro tipo.
Geralmente, eles são usados para descrever objetos utilizados
somente pelos tipos que os contêm.
Consulte também
Guia de Programação em C#
Classes
Classes e membros de classes abstract e sealed
(Guia de Programação em C#)
25/11/2019 • 4 minutes to read • Edit Online
A palavra-chave abstract permite que você crie classes e membros de classe que estão incompletos e devem
ser implementados em uma classe derivada.
A palavra-chave sealed permite evitar a herança de uma classe ou de determinados membros de classe que
foram marcados anteriormente com virtual.
Uma classe abstrata não pode ser instanciada. A finalidade de uma classe abstrata é fornecer uma definição
comum de uma classe base que pode ser compartilhada por várias classes derivadas. Por exemplo, uma
biblioteca de classes pode definir uma classe abstrata que serve como um parâmetro para muitas de suas
funções e exige que os programadores que usam essa biblioteca forneçam sua própria implementação da
classe, criando uma classe derivada.
As classes abstratas também podem definir métodos abstratos. Isso é realizado através da adição da palavra-
chave abstract antes do tipo de retorno do método. Por exemplo:
Os métodos abstratos não têm implementação, portanto, a definição do método é seguida por um ponto e
vírgula, em vez de um bloco de método normal. As classes derivadas da classe abstrata devem implementar
todos os métodos abstratos. Quando uma classe abstrata herda um método virtual de uma classe base, a
classe abstrata pode substituir o método virtual por um método abstrato. Por exemplo:
// compile with: -target:library
public class D
{
public virtual void DoWork(int i)
{
// Original implementation.
}
}
public class F : E
{
public override void DoWork(int i)
{
// New implementation.
}
}
Se um método virtual for declarado abstract , ele ainda será virtual para qualquer classe que herdar da
classe abstrata. Uma classe que herda um método abstrato não pode acessar a implementação original do
método. No exemplo anterior, DoWork na classe F não pode chamar DoWork na classe D. Dessa forma, uma
classe abstrata pode forçar classes derivadas a fornecerem novas implementações de método para métodos
virtuais.
Uma classe sealed não pode ser usada como uma classe base. Por esse motivo, também não pode ser uma
classe abstrata. As classes sealed impedem a derivação. Como elas nunca podem ser usadas como uma
classe base, algumas otimizações em tempo de execução podem tornar a chamada a membros de classe
sealed ligeiramente mais rápida.
Um método, um indexador, uma propriedade ou um evento em uma classe derivada que está substituindo
um membro virtual da classe base, pode declarar esse membro como sealed. Isso anula o aspecto virtual do
membro para qualquer outra classe derivada. Isso é realizado através da colocação da palavra-chave sealed
antes da palavra-chave override na declaração de membro de classe. Por exemplo:
public class D : C
{
public sealed override void DoWork() { }
}
Consulte também
Guia de Programação em C#
Classes e Structs
Herança
Métodos
Campos
Como definir propriedades abstratas
Classes static e membros de classes static (Guia de
Programação em C#)
04/11/2019 • 10 minutes to read • Edit Online
Uma classe static é basicamente o mesmo que uma classe não estática, mas há uma diferença: uma classe
estática não pode ser instanciada. Em outras palavras, você não pode usar o operador new para criar uma
variável do tipo de classe. Como não há nenhuma variável de instância, você acessa os membros de uma classe
estática usando o próprio nome de classe. Por exemplo, se houver uma classe estática chamada UtilityClass
com um método público chamado MethodA , chame o método, como mostra o exemplo a seguir:
UtilityClass.MethodA();
Uma classe estática pode ser usada como um contêiner conveniente para conjuntos de métodos que operam
apenas em parâmetros de entrada e não precisam obter ou definir campos de instância internos. Por exemplo,
na biblioteca de classes .NET Framework, a classe estática System.Math contém métodos que executam
operações matemáticas, sem a necessidade de armazenar ou recuperar dados que são exclusivos de uma
determinada instância da classe Math. Ou seja, você aplica os membros da classe especificando o nome de
classe e o nome do método, conforme mostrado no exemplo a seguir.
// Output:
// 3.14
// -4
// 3
Como é o caso com todos os tipos de classe, as informações de tipo de uma classe estática são carregadas pelo
CLR (Common Language Runtime) do .NET Framework quando o programa que faz referência à classe é
carregado. O programa não pode especificar exatamente quando a classe é carregada. No entanto, é garantido
que ela será carregada e terá seus campos inicializados e seu construtor estático chamado antes que a classe
seja referenciada pela primeira vez em seu programa. Um construtor estático é chamado apenas uma vez e
uma classe estática permanece na memória pelo tempo de vida do domínio do aplicativo em que seu
programa reside.
NOTE
Para criar uma classe não estática que permite que apenas uma instância de si mesma seja criada, consulte
Implementando singleton no C#.
Exemplo
Temos aqui um exemplo de uma classe estática que contém dois métodos que convertem a temperatura de
Celsius em Fahrenheit e de Fahrenheit em Celsius:
return fahrenheit;
}
return celsius;
}
}
class TestTemperatureConverter
{
static void Main()
{
Console.WriteLine("Please select the convertor direction");
Console.WriteLine("1. From Celsius to Fahrenheit.");
Console.WriteLine("2. From Fahrenheit to Celsius.");
Console.Write(":");
switch (selection)
{
case "1":
Console.Write("Please enter the Celsius temperature: ");
F = TemperatureConverter.CelsiusToFahrenheit(Console.ReadLine());
Console.WriteLine("Temperature in Fahrenheit: {0:F2}", F);
break;
case "2":
Console.Write("Please enter the Fahrenheit temperature: ");
C = TemperatureConverter.FahrenheitToCelsius(Console.ReadLine());
Console.WriteLine("Temperature in Celsius: {0:F2}", C);
break;
default:
Console.WriteLine("Please select a convertor.");
break;
}
Membros Estáticos
Uma classe não estática não pode conter métodos, campos, propriedades ou eventos estáticos. O membro
estático pode ser chamado em uma classe, mesmo quando nenhuma instância da classe foi criada. O membro
estático sempre é acessado pelo nome de classe, não pelo nome da instância. Existe apenas uma cópia de um
membro estático, independentemente de quantas instâncias da classe forem criadas. Propriedades e métodos
estáticos não podem acessar eventos e campos não estáticos no tipo que os contêm e não podem acessar uma
variável de instância de nenhum objeto, a menos que ele seja passado explicitamente em um parâmetro de
método.
É mais comum declarar uma classe não estática com alguns membros estáticos do que declarar uma classe
inteira como estática. Dois usos comuns dos campos estáticos são manter uma contagem do número de
objetos que foram instanciados ou armazenar um valor que deve ser compartilhado entre todas as instâncias.
Métodos estáticos podem ser sobrecarregados, mas não substituídos, porque pertencem à classe e não a
qualquer instância da classe.
Embora um campo não possa ser declarado como static const , um campo const é essencialmente estático
em seu comportamento. Ele pertence ao tipo e não a instâncias do tipo. Portanto, campos constantes podem
ser acessados usando a mesma notação ClassName.MemberName usada para campos estáticos. Nenhuma
instância de objeto é necessária.
O C# não dá suporte a variáveis locais estáticas (variáveis que são declaradas no escopo do método).
Você declara membros de classe estática usando a palavra-chave static antes do tipo de retorno do membro,
conforme mostrado no exemplo a seguir:
public class Automobile
{
public static int NumberOfWheels = 4;
public static int SizeOfGasTank
{
get
{
return 15;
}
}
public static void Drive() { }
public static event EventType RunOutOfGas;
Membros estáticos são inicializados antes que o membro estático seja acessado pela primeira vez e antes que
o construtor estático, se houver, seja chamado. Para acessar um membro de classe estática, use o nome da
classe em vez de um nome de variável para especificar o local do membro, conforme mostrado no exemplo a
seguir:
Automobile.Drive();
int i = Automobile.NumberOfWheels;
Se sua classe contiver campos estáticos, forneça um construtor estático que os inicializa quando a classe é
carregada.
Uma chamada para um método estático gera uma instrução de chamada em MSIL (Microsoft Intermediate
Language), enquanto uma chamada para um método de instância gera uma instrução callvirt , que também
verifica se há referências de objeto nulas. No entanto, na maioria das vezes, a diferença de desempenho entre
os dois não é significativa.
Especificação da Linguagem C#
Para saber mais, confira Classes estáticas e Membros estáticos e de instância na Especificação da linguagem
C#. A especificação da linguagem é a fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Guia de Programação em C#
static
Classes
class
Construtores estáticos
Construtores de instância
Modificadores de acesso (Guia de Programação
em C#)
23/10/2019 • 9 minutes to read • Edit Online
Todos os tipos e membros de tipo têm um nível de acessibilidade, que controla se podem ser usados de outro
código no seu assembly ou outros assemblies. Você pode usar os modificadores de acesso a seguir para
especificar a acessibilidade de um tipo ou membro quando você o declarar:
public
O tipo ou membro pode ser acessado por qualquer outro código no mesmo assembly ou em outro assembly
que faz referência a ele.
private
O tipo ou membro pode ser acessado somente pelo código na mesma classe ou struct.
protected
O tipo ou membro pode ser acessado somente pelo código na mesma classe ou em uma classe derivada
dessa classe.
internal
O tipo ou membro pode ser acessado por qualquer código no mesmo assembly, mas não de outro assembly.
protected internal O tipo ou membro pode ser acessado por qualquer código no assembly no qual ele é
declarado ou de uma classe derivada em outro assembly.
private protected O tipo ou membro pode ser acessado somente dentro de seu assembly de declaração, por
código na mesma classe ou em um tipo que é derivado dessa classe.
Os exemplos a seguir demonstram como especificar modificadores de acesso em um tipo e membro:
Nem todos os modificadores de acesso podem ser usados por todos os tipos ou membros em todos os
contextos e, em alguns casos, a acessibilidade de um membro de tipo é restrita pela acessibilidade do seu tipo
recipiente. As seções a seguir fornecem mais detalhes sobre a acessibilidade.
// public class:
public class Tricycle
{
// protected method:
protected void Pedal() { }
// private field:
private int wheels = 3;
NOTE
O nível de acessibilidade interno protegido significa protegido OU interno, não protegido E interno. Em outras palavras,
um membro interno protegido pode ser acessado de qualquer classe no mesmo assembly, incluindo as classes
derivadas. Para limitar a acessibilidade para apenas classes derivadas no mesmo assembly, declare a classe em si como
interna e declare seus membros como protegidos. Além disso, a partir do C# 7.2, você pode usar o modificador de
acesso protegido privado para obter o mesmo resultado sem a necessidade de tornar a classe de conteúdo interna.
Outros tipos
Interfaces declaradas diretamente dentro de um namespace podem ser declaradas como públicas ou internas
e, exatamente como as classes e structs, as interfaces assumem como padrão o acesso interno. Membros de
interface sempre são públicos, pois a finalidade de uma interface é permitir que outros tipos acessem uma
classe ou struct. Nenhum modificador de acesso pode ser aplicado aos membros de interface.
Membros de enumeração sempre são públicos e nenhum modificador de acesso pode ser aplicado.
Delegados se comportam como classes e structs. Por padrão, eles têm acesso interno quando declarados
diretamente dentro de um namespace e acesso privado quando aninhados.
Especificação da Linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a
fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Guia de Programação em C#
Classes e Structs
Interfaces
private
public
internal
protected
protected internal
private protected
class
struct
interface
Campos (Guia de Programação em C#)
25/11/2019 • 6 minutes to read • Edit Online
Um campo é uma variável de qualquer tipo que é declarada diretamente em uma classe ou struct. Os campos são
membros do tipo que os contém.
Uma classe ou um struct podem ter campos de instância, campos estáticos ou ambos. Os campos de instância
são específicos a uma instância de um tipo. Se você tem uma classe T, com um campo de instância F, você pode
criar dois objetos do tipo T e modificar o valor de F em cada objeto sem afetar o valor no outro objeto. Por outro
lado, um campo estático pertence à própria classe e é compartilhado entre todas as instâncias dessa classe. As
alterações feitas da instância A serão visíveis imediatamente para as instâncias B e C se acessarem o campo.
Em geral, você só deve usar campos para variáveis que têm acessibilidade particular ou protegida. Os dados que
a classe expõe para o código de cliente devem ser fornecidos por meio de métodos, propriedades e indexadores.
Usando esses constructos para acesso indireto aos campos internos, você pode proteger contra valores de
entrada inválidos. Um campo particular que armazena os dados expostos por uma propriedade pública é
chamado de repositório de backup ou de campo de suporte.
Os campos normalmente armazenam os dados que devem estar acessíveis a mais de um método de classe e
devem ser armazenados por mais tempo que o tempo de vida de qualquer método único. Por exemplo, uma
classe que representa uma data do calendário pode ter três campos de inteiros: um para o mês, um para o dia e
outro para o ano. As variáveis que não são usadas fora do escopo de um método único devem ser declaradas
como variáveis locais dentro do próprio corpo do método.
Os campos são declarados no bloco de classe, especificando o nível de acesso do campo, seguido pelo tipo do
campo e pelo nome do campo. Por exemplo:
public class CalendarEntry
{
// private field
private DateTime date;
}
}
Para acessar um campo em um objeto, adicione um ponto após o nome do objeto, seguido pelo nome do campo,
como em objectname.fieldname . Por exemplo:
Um campo pode receber um valor inicial, usando o operador de atribuição quando o campo é declarado. Para
atribuir automaticamente o campo day ao "Monday" , por exemplo, você poderia declarar day como no exemplo
a seguir:
Os campos são inicializados imediatamente antes do construtor para a instância do objeto ser chamado. Se o
construtor atribuir o valor de um campo, ele substituirá qualquer valor fornecido durante a declaração do campo.
Para obter mais informações, veja Usando construtores.
NOTE
Um inicializador de campo não pode fazer referência a outros campos de instância.
Os campos podem ser marcados como public, private, protected, internal, protected internal ou private protected.
Esses modificadores de acesso definem como os usuários da classe podem acessar os campos. Para obter mais
informações, consulte Modificadores de acesso.
Opcionalmente, um campo pode ser declarado static. Isso torna o campo disponível para chamadores a qualquer
momento, mesmo se não existir nenhuma instância da classe. Para obter mais informações, consulte Classes
Estáticas e Membros de Classes Estáticas.
Um campo pode ser declarado readonly. Um valor só pode ser atribuído a um campo somente leitura durante a
inicialização ou em um construtor. Um campo static readonly é muito semelhante a uma constante, exceto que
o compilador C# não tem acesso ao valor de um campo somente leitura estático em tempo de compilação, mas
somente em tempo de execução. Para obter mais informações, consulte Constantes.
Especificação da Linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a fonte
definitiva para a sintaxe e o uso de C#.
Consulte também
Guia de Programação em C#
Classes e Structs
Usando construtores
Herança
Modificadores de acesso
Classes e membros de classes abstract e sealed
Constantes (Guia de Programação em C#)
31/10/2019 • 4 minutes to read • Edit Online
As constantes são valores imutáveis que são conhecidos no tempo de compilação e não são alterados durante a
vida útil do programa. Constantes são declaradas com o modificador const. Apenas os tipos C# internos
(excluindo System.Object) podem ser declarados como const . Para obter uma lista dos tipos internos, consulte
Tabela de tipos internos. Tipos definidos pelo usuário, incluindo classes, struct e matrizes, não podem ser const .
Use o modificador readonly para criar uma classe, struct ou matriz que é inicializada uma vez em tempo de
execução (por exemplo, em um construtor) e, assim, não pode ser alterada.
O C# não dá suporte aos métodos const , propriedades ou eventos.
O tipo de enumeração permite que você defina constantes nomeadas para tipos internos integrais (por exemplo
int , uint , long e assim por diante). Para obter mais informações, consulte enum.
As constantes devem ser inicializadas conforme elas são declaradas. Por exemplo:
class Calendar1
{
public const int Months = 12;
}
Neste exemplo, a constante months sempre é 12 e não pode ser alterada até mesmo pela própria classe. Na
verdade, quando o compilador encontra um identificador constante no código-fonte C# (por exemplo, months ),
ele substitui o valor literal diretamente no código de IL (linguagem intermediária) que ele produz. Como não há
nenhum endereço variável associado a uma constante em tempo de execução, os campos const não podem ser
passados por referência e não podem aparecer como um l-value em uma expressão.
NOTE
Tenha cuidado ao fazer referência a valores constantes definidos em outro código como DLLs. Se uma nova versão da DLL
definir um novo valor para a constante, seu programa ainda conterá o valor literal antigo até que ele seja recompilado com
a nova versão.
Várias constantes do mesmo tipo podem ser declaradas ao mesmo tempo, por exemplo:
class Calendar2
{
public const int Months = 12, Weeks = 52, Days = 365;
}
A expressão que é usada para inicializar uma constante poderá fazer referência a outra constante se ela não criar
uma referência circular. Por exemplo:
class Calendar3
{
public const int Months = 12;
public const int Weeks = 52;
public const int Days = 365;
As constantes podem ser marcadas como public, private, protected, internal, protected internal ou private
protected. Esses modificadores de acesso definem como os usuários da classe podem acessar a constante. Para
obter mais informações, consulte Modificadores de Acesso.
As constantes são acessadas como se fossem campos static porque o valor da constante é o mesmo para todas
as instâncias do tipo. Você não usa a palavra-chave static para declará-las. As expressões que não estão na
classe que define a constante devem usar o nome de classe, um período e o nome da constante para acessar a
constante. Por exemplo:
Especificação da Linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a fonte
definitiva para a sintaxe e o uso de C#.
Consulte também
Guia de Programação em C#
Classes e Structs
Propriedades
Tipos
readonly
Imutabilidade em C# parte um: tipos de imutabilidade
Como definir propriedades abstratas (C# guia de
programação)
25/11/2019 • 3 minutes to read • Edit Online
O exemplo a seguir mostra como definir propriedades abstract. Uma declaração de propriedade abstract não
fornece uma implementação dos acessadores da propriedade – ela declara que a classe dá suporte às
propriedades, mas deixa a implementação do acessador para classes derivadas. O exemplo a seguir demonstra
como implementar as propriedades abstract herdadas de uma classe base.
Esse exemplo consiste em três arquivos, cada um deles é compilado individualmente e seu assembly resultante é
referenciado pela próxima compilação:
abstractshape.cs: a classe Shape que contém uma propriedade abstract Area .
shapes.cs: as subclasses da classe Shape .
shapetest.cs: um programa de teste para exibir as áreas de alguns objetos derivados de Shape .
Para compilar o exemplo, use o comando a seguir:
csc abstractshape.cs shapes.cs shapetest.cs
Exemplo
Esse arquivo declara a classe Shape que contém a propriedade Area do tipo double .
// compile with: csc -target:library abstractshape.cs
public abstract class Shape
{
private string name;
public Shape(string s)
{
// calling the set accessor of the Id property.
Id = s;
}
public string Id
{
get
{
return name;
}
set
{
name = value;
}
}
Ao declarar uma propriedade abstract (como Area neste exemplo), você simplesmente indica quais
acessadores de propriedade estão disponíveis, mas não os implementa. Neste exemplo, apenas um
acessador get está disponível, assim, a propriedade é somente leitura.
Exemplo
O código a seguir mostra três subclasses de Shape e como elas substituem a propriedade Area para fornecer
sua própria implementação.
// compile with: csc -target:library -reference:abstractshape.dll shapes.cs
public class Square : Shape
{
private int side;
Exemplo
O código a seguir mostra um programa de teste que cria uma quantidade de objetos derivados de Shape e
imprime suas áreas.
System.Console.WriteLine("Shapes Collection");
foreach (Shape s in shapes)
{
System.Console.WriteLine(s);
}
}
}
/* Output:
Shapes Collection
Square #1 Area = 25.00
Circle #1 Area = 28.27
Rectangle #1 Area = 20.00
*/
Consulte também
Guia de Programação em C#
Classes e Structs
Classes e membros de classes abstract e sealed
Propriedades
Como definir constantes em C#
25/11/2019 • 2 minutes to read • Edit Online
As constantes são campos cujos valores são definidos em tempo de compilação e nunca podem ser alterados. Use
constantes para fornecer nomes significativos em vez de literais numéricos ("números mágicos") a valores
especiais.
NOTE
No C#, a diretiva de pré-processador #define não pode ser utilizada para definir constantes da mesma maneira que é
normalmente usada no C e no C++.
Para definir valores de constantes de tipos integrais ( int , byte e assim por diante), use um tipo enumerado. Para
obter mais informações, consulte enum.
Para definir constantes não integrais, uma abordagem é agrupá-las em uma única classe estática de nome
Constants . Isso exigirá que todas as referências às constantes sejam precedidas com o nome de classe, conforme
mostrado no exemplo a seguir.
Exemplo
static class Constants
{
public const double Pi = 3.14159;
public const int SpeedOfLight = 300000; // km per sec.
}
class Program
{
static void Main()
{
double radius = 5.3;
double area = Constants.Pi * (radius * radius);
int secsFromSun = 149476000 / Constants.SpeedOfLight; // in km
}
}
O uso do qualificador de nome de classe ajuda a garantir que você e outras pessoas que usam a constante
entendam que ele é constante e não pode ser modificado.
Consulte também
Classes e Structs
Propriedades (Guia de Programação em C#)
04/11/2019 • 8 minutes to read • Edit Online
Uma propriedade é um membro que oferece um mecanismo flexível para ler, gravar ou calcular o valor
de um campo particular. As propriedades podem ser usadas como se fossem membros de dados
públicos, mas na verdade elas são métodos realmente especiais chamados acessadores. Isso permite que
os dados sejam acessados facilmente e ainda ajuda a promover a segurança e a flexibilidade dos
métodos.
class TimePeriod
{
private double _seconds;
class Program
{
static void Main()
{
TimePeriod t = new TimePeriod();
// The property assignment causes the 'set' accessor to be called.
t.Hours = 24;
Começando com o C# 7.0, os acessadores get e set podem ser implementados como membros aptos
para expressão. Nesse caso, as palavras-chave get e set devem estar presentes. O exemplo a seguir
ilustra o uso de definições de corpo de expressão para ambos os acessadores. Observe que a palavra-
chave return não é usada com o acessador get .
using System;
class Program
{
static void Main(string[] args)
{
var item = new SaleItem("Shoes", 19.95m);
Console.WriteLine($"{item.Name}: sells for {item.Price:C2}");
}
}
// The example displays output like the following:
// Shoes: sells for $19.95
Propriedades autoimplementadas
Em alguns casos, os acessadores get e set da propriedade apenas atribuem um valor ou recuperam
um valor de um campo de suporte sem incluir nenhuma lógica adicional. Usando propriedades
autoimplementadas, você pode simplificar o código enquanto o compilador C# fornece de forma
transparente o campo de suporte para você.
Se uma propriedade tiver tanto um acessador get quanto um set , ambos deverão ser
autoimplementados. Você define uma propriedade autoimplementada usando as palavras-chave get e
set sem fornecer qualquer implementação. O exemplo a seguir repete o anterior, exceto que Name e
Price são propriedades autoimplementadas. Observe que o exemplo também remove o construtor com
parâmetros, de modo que objetos SaleItem agora são inicializados com uma chamada para o construtor
sem parâmetros e um inicializador de objeto.
using System;
class Program
{
static void Main(string[] args)
{
var item = new SaleItem{ Name = "Shoes", Price = 19.95m };
Console.WriteLine($"{item.Name}: sells for {item.Price:C2}");
}
}
// The example displays output like the following:
// Shoes: sells for $19.95
Seções relacionadas
Usando propriedades
Propriedades de interface
Comparação entre propriedades e indexadores
Restringindo a acessibilidade ao acessador
Propriedades Autoimplementadas
Especificação da Linguagem C#
Para obter mais informações, veja Propriedades na Especificação da Linguagem C#. A especificação da
linguagem é a fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Guia de Programação em C#
Usando propriedades
Indexadores
Palavra-chave get
Palavra-chave set
Usando propriedades (Guia de Programação em C#)
23/10/2019 • 14 minutes to read • Edit Online
As propriedades combinam aspectos de métodos e campos. Para o usuário de um objeto, uma propriedade
parece ser um campo. Acessar a propriedade requer a mesma sintaxe. Para o implementador de uma classe, uma
propriedade consiste em um ou dois blocos de código, que representam um acessador get e/ou um acessador set.
O bloco de código para o acessador get é executado quando a propriedade é lida. O bloco de código para o
acessador set é executado quando um novo valor é atribuído à propriedade. Uma propriedade sem um
acessador set é considerada como somente leitura. Uma propriedade sem um acessador get é considerada
como somente gravação. Uma propriedade que tem os dois acessadores é leitura/gravação.
Diferentemente dos campos, as propriedades não são classificadas como variáveis. Portanto, você não pode
passar uma propriedade como um parâmetro ref ou out.
As propriedades têm muitos usos: elas podem validar os dados antes de permitir uma alteração; elas podem
expor, de forma transparente, os dados em uma classe em que esses dados são realmente recuperados de outra
origem qualquer, como um banco de dados; elas podem executar uma ação quando os dados são alterados, como
acionar um evento ou alterar o valor de outros campos.
As propriedades são declaradas no bloco de classe, especificando o nível de acesso do campo, seguido pelo tipo
da propriedade, pelo nome da propriedade e por um bloco de código que declara um acessador get e/ou um
acessador set . Por exemplo:
Neste exemplo, Month é declarado como uma propriedade de maneira que o acessador set possa garantir que
o valor Month esteja definido entre 1 e 12. A propriedade Month usa um campo particular para rastrear o valor
real. O local real dos dados de uma propriedade é conhecido, em geral, como o "repositório de backup" da
propriedade. É comum as propriedades usarem campos particulares como um repositório de backup. O campo é
marcado como particular para garantir que ele só pode ser alterado ao chamar a propriedade. Para obter mais
informações sobre as restrições de acesso público e particular, consulte Modificadores de acesso.
As propriedades autoimplementadas fornecem sintaxe simplificada para declarações de propriedade simples.
Para obter mais informações, consulte Propriedades autoimplementadas.
O acessador get
O corpo do acessador get assemelha-se ao de um método. Ele deve retornar um valor do tipo de propriedade. A
execução do acessador get é equivalente à leitura do valor do campo. Por exemplo, quando você estiver
retornando a variável particular do acessador get e as otimizações estiverem habilitadas, a chamada ao método
do acessador get será embutida pelo compilador, portanto, não haverá nenhuma sobrecarga de chamada de
método. No entanto, um método do acessador get virtual não pode ser embutido porque o compilador não
sabe, em tempo de compilação, qual método pode ser chamado em tempo de execução. A seguir está um
acessador get que retorna o valor de um campo particular _name :
class Person
{
private string _name; // the name field
public string Name => _name; // the Name property
}
Quando você referencia a propriedade, exceto como o destino de uma atribuição, o acessador get é invocado
para ler o valor da propriedade. Por exemplo:
O acessador get deve terminar em uma instrução return ou throw e o controle não pode fluir para fora do corpo
do acessador.
Alterar o estado do objeto usando o acessador get é um estilo ruim de programação. Por exemplo, o acessador a
seguir produz o efeito colateral de alterar o estado do objeto sempre que o campo _number é acessado.
O acessador get pode ser usado para retornar o valor do campo ou para calculá-lo e retorná-lo. Por exemplo:
class Employee
{
private string _name;
public string Name => _name != null ? _name : "NA";
}
No segmento de código anterior, se você não atribuir um valor à propriedade Name , ele retornará o valor NA .
O acessador set
O acessador set é semelhante a um método cujo tipo de retorno é void. Ele usa uma parâmetro implícito
chamado value , cujo tipo é o tipo da propriedade. No exemplo a seguir, uma acessador set é adicionado à
propriedade Name :
class Person
{
private string _name; // the name field
public string Name // the Name property
{
get => _name;
set => _name = value;
}
}
Quando você atribui um valor à propriedade, o acessador set é invocado por meio do uso de um argumento
que fornece o novo valor. Por exemplo:
É um erro usar o nome de parâmetro implícito value , para uma declaração de variável local em um acessador
set .
Comentários
As propriedades podem ser marcadas como public , private , protected , internal , protected internal ou
private protected . Esses modificadores de acesso definem como os usuários da classe podem acessar a
propriedade. Os acessadores get e set para a mesma propriedade podem ter modificadores de acesso
diferentes. Por exemplo, o get pode ser public para permitir acesso somente leitura de fora do tipo e o set
pode ser private ou protected . Para obter mais informações, consulte Modificadores de acesso.
Uma propriedade pode ser declarada como uma propriedade estática, usando a palavra-chave static . Isso torna
a propriedade disponível para chamadores a qualquer momento, mesmo se não existir nenhuma instância da
classe. Para obter mais informações, consulte Classes Estáticas e Membros de Classes Estáticas.
Uma propriedade pode ser marcada como uma propriedade virtual, usando a palavra-chave virtual. Isso habilita
as classes derivadas a substituírem o comportamento da propriedade, usando a palavra-chave override. Para
obter mais informações sobre essas opções, consulte Herança.
Uma propriedade que substitui uma propriedade virtual também pode ser sealed, especificando que ela não é
mais virtual para classes derivadas. Por fim, uma propriedade pode ser declarada abstract. Isso significa que não
há nenhuma implementação na classe e as classes derivadas devem escrever sua própria implementação. Para
obter mais informações sobre essas opções, consulte Classes e membros de classes abstract e sealed.
NOTE
É um erro usar um modificador virtual, abstract ou override em um acessador de uma propriedade static.
Exemplo
Este exemplo demonstra as propriedades instância, estática e somente leitura. Ele aceita o nome do funcionário
digitado no teclado, incrementa NumberOfEmployees em 1 e exibe o nome e o número do funcionário.
public class Employee
{
public static int NumberOfEmployees;
private static int _counter;
private string _name;
// A Constructor:
public Employee() => _counter = ++NumberOfEmployees; // Calculate the employee's number:
}
class TestEmployee
{
static void Main()
{
Employee.NumberOfEmployees = 107;
Employee e1 = new Employee();
e1.Name = "Claude Vige";
Exemplo
Este exemplo demonstra como acessar uma propriedade em uma classe base que é ocultada por outra
propriedade que tem o mesmo nome em uma classe derivada:
public class Employee
{
private string _name;
public string Name
{
get => _name;
set => _name = value;
}
}
class TestHiding
{
static void Main()
{
Manager m1 = new Manager();
((Employee)m1).Name = "Mary";
Para obter mais informações sobre como ocultar membros, consulte o Modificador new.
Exemplo
Neste exemplo, duas classes, Cube e Square , implementam uma classe abstrata Shape e substituem sua
propriedade Area abstrata. Observe o uso do modificador override nas propriedades. O programa aceita o lado
como uma entrada e calcula as áreas para o cubo e o quadrado. Ele também aceita a área como uma entrada e
calcula o lado correspondente para o cubo e o quadrado.
//constructor
public Square(double s) => side = s;
//constructor
public Cube(double s) => side = s;
class TestShapes
{
static void Main()
{
// Input the side:
System.Console.Write("Enter the side: ");
double side = double.Parse(System.Console.ReadLine());
Consulte também
Guia de Programação em C#
Propriedades
Propriedades de interface
Propriedades Autoimplementadas
Propriedades de interface (Guia de Programação em
C#)
23/10/2019 • 2 minutes to read • Edit Online
As propriedades podem ser declaradas em uma interface. Este é um exemplo de um acessador de propriedade de
interface:
O acessador de uma propriedade de interface não tem um corpo. Portanto, a finalidade dos acessadores é indicar
se a propriedade é leitura/gravação, somente leitura ou somente gravação.
Exemplo
Neste exemplo, a interface IEmployee tem uma propriedade de leitura/gravação, Name e uma propriedade
somente leitura, Counter . A classe Employee implementa a interface IEmployee e usa essas duas propriedades.
O programa lê o nome de um novo funcionário e o número atual de funcionários e exibe o nome do funcionário
e o número do funcionário computado.
Seria possível usar o nome totalmente qualificado da propriedade, que referencia a interface na qual o membro é
declarado. Por exemplo:
string IEmployee.Name
{
get { return "Employee Name"; }
set { }
}
Isso é denominado Implementação explícita da interface. Por exemplo, se a classe Employee estiver
implementando duas interfaces ICitizen e IEmployee e as duas interfaces tiverem a propriedade Name , será
necessária a implementação explícita de membro da interface. Ou seja, a seguinte declaração de propriedade:
string IEmployee.Name
{
get { return "Employee Name"; }
set { }
}
interface IEmployee
{
string Name
{
get;
set;
}
int Counter
{
get;
}
}
// constructor
public Employee() => _counter = ++numberOfEmployees;
}
class TestEmployee
{
static void Main()
{
System.Console.Write("Enter number of employees: ");
Employee.numberOfEmployees = int.Parse(System.Console.ReadLine());
Saída de exemplo
Enter number of employees: 210
Enter the name of the new employee: Hazem Abolrous
The employee information:
Employee number: 211
Employee name: Hazem Abolrous
Consulte também
Guia de Programação em C#
Propriedades
Usando propriedades
Comparação entre propriedades e indexadores
Indexadores
Interfaces
Restringindo a acessibilidade ao acessador (Guia de
Programação em C#)
23/10/2019 • 7 minutes to read • Edit Online
As partes get e set de uma propriedade ou de um indexador são chamadas acessadores. Por padrão, esses
acessadores têm a mesma visibilidade ou nível de acesso da propriedade ou do indexador aos quais pertencem.
Para obter mais informações, consulte níveis de acessibilidade. No entanto, às vezes é útil restringir o acesso a um
desses acessadores. Normalmente, isso envolve restringir a acessibilidade do acessador set e manter o
acessador get publicamente acessível. Por exemplo:
Neste exemplo, uma propriedade chamada Name define um acessador get e set . O acessador get recebe o
nível de acessibilidade da propriedade em si, public nesse caso, embora o set acessador esteja restrito
explicitamente ao aplicar o modificador de acesso protegido ao acessador em si.
Implementando interfaces
Quando você usa um acessador para implementar uma interface, o acessador pode não ter um modificador de
acesso. No entanto, se você implementar a interface usando um acessador, como get , o outro acessador poderá
ter um modificador de acesso, como no exemplo a seguir:
public string Id
{
get { return _id; }
set { }
}
}
class MainClass
{
static void Main()
{
BaseClass b1 = new BaseClass();
DerivedClass d1 = new DerivedClass();
b1.Name = "Mary";
d1.Name = "John";
b1.Id = "Mary123";
d1.Id = "John123"; // The BaseClass.Id property is called.
Comentários
Observe que, se você substituir a declaração new private string Id por new public string Id , você obterá a
saída:
Name and ID in the base class: Name-BaseClass, ID-BaseClass
Consulte também
Guia de Programação em C#
Propriedades
Indexadores
Modificadores de acesso
Como declarar e usar as propriedades de leitura/C#
gravação (guia de programação)
25/11/2019 • 4 minutes to read • Edit Online
As propriedades oferecem a conveniência de membros de dados públicos sem os riscos associados ao acesso sem
proteção, sem controle e não verificado aos dados de um objeto. Isso é feito por meio de acessadores: métodos
especiais que atribuem e recuperam valores do membro de dados subjacente. O acessador set habilita a atribuição
de membros de dados e o acessador get recupera valores do membro de dados.
Este exemplo mostra uma classe Person que tem duas propriedades: Name (string) e Age (int). Ambas as
propriedades fornecem acessadores get e set , portanto, são consideradas propriedades de leitura/gravação.
Exemplo
class Person
{
private string _name = "N/A";
private int _age = 0;
set
{
_age = value;
}
}
class TestPerson
{
static void Main()
{
// Create a new Person object:
Person person = new Person();
// Print out the name and the age associated with the person:
Console.WriteLine("Person details - {0}", person);
Programação robusta
No exemplo anterior, as propriedades Name e Age são públicas e incluem os acessadores get e set . Isso
permite que qualquer objeto leia e grave essas propriedades. No entanto, às vezes é desejável excluir um os
acessadores. Omitir o acessador set , por exemplo, torna a propriedade somente leitura:
Como alternativa, é possível expor um acessador publicamente, porém, tornando o outro privado ou protegido.
Para obter mais informações, consulte Acessibilidade do Acessador Assimétrico.
Depois de serem declaradas, as propriedades podem ser usadas como campos da classe. Isso permite uma sintaxe
muito natural na obtenção e configuração do valor de uma propriedade, conforme as instruções a seguir:
person.Name = "Joe";
person.Age = 99;
Observe que em um método de propriedade set , uma variável especial value está disponível. Essa variável
contém o valor que o usuário especificou, por exemplo:
_name = value;
person.Age += 1;
Se métodos set e get separados fossem usados para modelar propriedades, o código equivalente se pareceria
com isto:
person.SetAge(person.GetAge() + 1);
Observe que ToString não é usado explicitamente no programa. Ele é invocado por padrão pelas chamadas
WriteLine .
Consulte também
Guia de Programação em C#
Propriedades
Classes e Structs
Propriedades autoimplementadas (Guia de
Programação em C#)
25/11/2019 • 2 minutes to read • Edit Online
Exemplo
O exemplo a seguir mostra uma classe simples que tem algumas propriedades autoimplementadas:
// Constructor
public Customer(double purchases, string name, int ID)
{
TotalPurchases = purchases;
Name = name;
CustomerID = ID;
}
// Methods
public string GetContactInfo() { return "ContactInfo"; }
public string GetTransactionHistory() { return "History"; }
class Program
{
static void Main()
{
// Intialize a new object.
Customer cust1 = new Customer(4987.63, "Northwind", 90108);
// Modify a property.
cust1.TotalPurchases += 499.99;
}
}
Consulte também
Propriedades
Modificadores
Como implementar uma classe leve com
propriedades implementadas automaticamente (C#
guia de programação)
25/11/2019 • 3 minutes to read • Edit Online
Este exemplo mostra como criar uma classe leve imutável que serve apenas para encapsular um conjunto de
propriedades autoimplementadas. Use esse tipo de constructo em vez de um struct quando for necessário usar a
semântica do tipo de referência.
É possível tornar uma propriedade imutável de duas maneiras:
É possível declarar o acessador set como privado. A propriedade será configurável somente dentro do tipo,
mas será imutável para os consumidores.
Ao declarar um acessador privado set , não é possível usar um inicializador de objeto para inicializar a
propriedade. É necessário usar um construtor ou um método de fábrica.
É possível declarar somente o acessador get, o que torna a propriedade imutável em todos os lugares,
exceto no construtor do tipo.
Exemplo
O exemplo a seguir mostra duas maneiras de implementar uma classe imutável que tem propriedades
autoimplementadas. Entre essas maneiras, uma declara uma das propriedades com um set privado e outra
declara uma das propriedades somente com um get . A primeira classe usa um construtor somente para
inicializar as propriedades e a segunda classe usa um método de fábrica estático que chama um construtor.
// Public constructor.
public Contact(string contactName, string contactAddress)
{
Name = contactName;
Address = contactAddress;
}
}
// Private constructor.
private Contact2(string contactName, string contactAddress)
{
{
Name = contactName;
Address = contactAddress;
}
/* Output:
Terry Adams, 123 Main St.
Fadi Fakhouri, 345 Cypress Ave.
Hanying Feng, 678 1st Ave
Cesar Garcia, 12 108th St.
Debra Garcia, 89 E. 42nd St.
*/
O compilador cria campos de suporte para cada propriedade autoimplementada. Os campos não são acessíveis
diretamente do código-fonte.
Consulte também
Propriedades
struct
Inicializadores de objeto e coleção
Métodos (Guia de Programação em C#)
23/10/2019 • 18 minutes to read • Edit Online
Um método é um bloco de código que contém uma série de instruções. Um programa faz com que as
instruções sejam executadas chamando o método e especificando os argumentos de método necessários.
No C#, todas as instruções executadas são realizadas no contexto de um método. O método Main é o
ponto de entrada para C# cada aplicativo e é chamado pelo Common Language Runtime (CLR ) quando o
programa é iniciado.
NOTE
Este artigo discute os métodos nomeados. Para obter informações sobre funções anônimas, consulte Funções
anônimas.
Assinaturas de método
Os métodos são declarados em uma classe ou struct especificando o nível de acesso, como public ou
private , modificadores opcionais, como abstract ou sealed , o valor retornado, o nome do método e os
parâmetros de método. Juntas, essas partes são a assinatura do método.
NOTE
Um tipo de retorno de um método não faz parte da assinatura do método para fins de sobrecarga de método. No
entanto, ele faz parte da assinatura do método ao determinar a compatibilidade entre um delegado e o método
para o qual ele aponta.
Os parâmetros de método estão entre parênteses e separados por vírgulas. Parênteses vazios indicam que
o método não requer parâmetros. Essa classe contém quatro métodos:
Acesso de método
Chamar um método em um objeto é como acessar um campo. Após o nome do objeto, adicione um ponto
final, o nome do método e parênteses. Os argumentos são listados dentro dos parênteses e são separados
por vírgulas. Os métodos da classe Motorcycle podem, portanto, ser chamados como no exemplo a
seguir:
class TestMotorcycle : Motorcycle
{
moto.StartEngine();
moto.AddGas(15);
moto.Drive(5, 20);
double speed = moto.GetTopSpeed();
Console.WriteLine("My top speed is {0}", speed);
}
}
int Square(int i)
{
// Store input argument in a local variable.
int input = i;
return input * input;
}
Agora, se você passar um objeto com base nesse tipo para um método, uma referência ao objeto será
passada. O exemplo a seguir passa um objeto do tipo SampleRefType para o método ModifyObject :
O exemplo faz essencialmente a mesma coisa que o exemplo anterior, pois ele passa um argumento por
valor para um método. No entanto, como um tipo de referência é usado, o resultado é diferente. A
modificação feita em ModifyObject para o campo value do parâmetro, obj , também altera o campo
value do argumento, rt , no método TestRefType . O método TestRefType exibe 33 como a saída.
Para obter mais informações sobre como passar tipos de referência por referência e por valor, consulte
Passando parâmetros de tipo de referência e Tipos de referência.
Valores de retorno
Os métodos podem retornar um valor para o chamador. Se o tipo de retorno, o tipo listado antes do nome
do método, não for void , o método poderá retornar o valor usando a palavra-chave return . Uma
instrução com a palavra-chave return seguida por uma variável que corresponde ao tipo de retorno
retornará esse valor ao chamador do método.
O valor pode ser retornado ao chamador por valor ou, começando com o C# 7.0, por referência. Valores
são retornados ao chamador por referência se a ref palavra-chave é usada na assinatura do método e
segue cada palavra-chave return . Por exemplo, a instrução de retorno e a assinatura de método a seguir
indicam que o método retorna uma variável chamada estDistance por referência para o chamador.
A palavra-chave return também interrompe a execução do método. Se o tipo de retorno for void , uma
instrução return sem um valor ainda será útil para interromper a execução do método. Sem a palavra-
chave return , a execução do método será interrompida quando chegar ao final do bloco de código.
Métodos com um tipo de retorno não nulo devem usar a palavra-chave return para retornar um valor.
Por exemplo, esses dois métodos usam a palavra-chave return para retornar inteiros:
class SimpleMath
{
public int AddTwoNumbers(int number1, int number2)
{
return number1 + number2;
}
Para usar um valor retornado de um método, o método de chamada pode usar a chamada de método em
si em qualquer lugar que um valor do mesmo tipo seria suficiente. Você também pode atribuir o valor
retornado a uma variável. Por exemplo, os dois exemplos de código a seguir obtêm a mesma meta:
Usar uma variável local, nesse caso, result , para armazenar um valor é opcional. Isso pode ajudar a
legibilidade do código ou pode ser necessário se você precisar armazenar o valor original do argumento
para todo o escopo do método.
Para usar o valor retornado de um método por referência, você deve declarar uma variável ref local se você
pretende modificar seu valor. Por exemplo, se o método Planet.GetEstimatedDistance retorna um valor
Double por referência, você pode defini-lo como uma variável ref local com código semelhante ao
seguinte:
Retornar uma matriz multidimensional de um método, M , que modifica o conteúdo da matriz, não é
necessário se a função de chamada passou a matriz para M . Você pode retornar a matriz resultante de M
para um bom estilo ou fluxo funcional de valores, mas isso não é necessário porque o C# passa todos os
tipos de referência por valor e o valor de uma referência de matriz é o ponteiro para a matriz. No método
M , qualquer alteração no conteúdo da matriz é observável por qualquer código que tenha uma referência
à matriz, conforme mostrado no exemplo a seguir:
static void Main(string[] args)
{
int[,] matrix = new int[2, 2];
FillMatrix(matrix);
// matrix is now full of -1
}
Métodos assíncronos
Usando o recurso async, você pode invocar métodos assíncronos sem usar retornos de chamada explícitos
ou dividir manualmente seu código entre vários métodos ou expressões lambda.
Se marcar um método com o modificador async, você poderá usar o operador await no método. Quando o
controle atinge uma expressão await no método assíncrono, ele retorna para o chamador e o progresso no
método é suspenso até a tarefa aguardada ser concluída. Quando a tarefa for concluída, a execução poderá
ser retomada no método.
NOTE
Um método assíncrono retorna para o chamador quando encontra o primeiro objeto esperado que ainda não está
completo ou chega ao final do método assíncrono, o que ocorrer primeiro.
Um método assíncrono pode conter um tipo de retorno Task<TResult>, Task ou nulo. O tipo de retorno
nulo é usado principalmente para definir manipuladores de eventos, em que um tipo de retorno nulo é
necessário. Um método assíncrono que retorna nulo não pode ser aguardado e o chamador de um método
de retorno nulo não pode capturar as exceções que esse método gera.
No exemplo a seguir, DelayAsync é um método assíncrono que tem um tipo de retorno de Task<TResult>.
DelayAsync tem uma instrução return que retorna um número inteiro. Portanto, a declaração do método
de Task<int> deve ter um tipo de retorno de DelayAsync . Como o tipo de retorno é Task<int> , a
avaliação da expressão await em DoSomethingAsync produz um inteiro, como a instrução a seguir
demonstra: int result = await delayTask .
O método startButton_Click é um exemplo de método assíncrono que tem um tipo de retorno nulo.
Como DoSomethingAsync é um método assíncrono, a tarefa para a chamada para DoSomethingAsync deve
ser colocada em espera, como mostra a seguinte instrução: await DoSomethingAsync(); . O método
startButton_Click deve ser definido com o modificador async porque o método tem uma expressão
await .
// using System.Diagnostics;
// using System.Threading.Tasks;
// Output:
// Result: 5
Um método assíncrono não pode declarar nenhum parâmetro ref ou out, mas pode chamar métodos com
tais parâmetros.
Para obter mais informações sobre os métodos assíncronos, consulte Programação assíncrona com async
e await, Fluxo de controle em programas assíncronos e Tipos de retorno assíncronos.
public Point Move(int dx, int dy) => new Point(x + dx, y + dy);
public void Print() => Console.WriteLine(First + " " + Last);
// Works with operators, properties, and indexers too.
public static Complex operator +(Complex a, Complex b) => a.Add(b);
public string Name => First + " " + Last;
public Customer this[long id] => store.LookupCustomer(id);
Se o método retornar void ou for um método assíncrono, o corpo do método deverá ser uma expressão
de instrução (igual às lambdas). Para propriedades e indexadores, eles devem ser somente leitura e você
não usa a palavra-chave do acessador get .
Iterators
Um iterador realiza uma iteração personalizada em uma coleção, como uma lista ou uma matriz. Um
iterador usa a instrução yield return para retornar um elemento de cada vez. Quando uma instrução yield
return for alcançada, o local atual no código será lembrado. A execução será reiniciada desse local quando
o iterador for chamado na próxima vez.
Você chama um iterador de um código de cliente usando uma instrução foreach.
O tipo de retorno de um iterador pode ser IEnumerable, IEnumerable<T>, IEnumerator ou
IEnumerator<T>.
Para obter mais informações, consulte Iteradores.
Especificação da linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a
fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Guia de Programação em C#
Classes e Structs
Modificadores de acesso
Classes static e membros de classes static
Herança
Classes e membros de classes abstract e sealed
params
return
out
ref
Passando parâmetros
Funções locais (Guia de Programação em C#)
08/11/2019 • 9 minutes to read • Edit Online
Começando com o C# 7.0, o C# é compatível com funções locais. Funções locais são métodos privados de um tipo
que estão aninhados em outro membro. Eles só podem ser chamados do membro que os contém. Funções locais
podem ser declaradas em e chamadas de:
Métodos, especialmente os métodos iteradores e os métodos assíncronos
Construtores
Acessadores de propriedade
Acessadores de eventos
Métodos anônimos
Expressões lambda
Finalizadores
Outras funções locais
No entanto, as funções locais não podem ser declaradas dentro de um membro apto para expressão.
NOTE
Em alguns casos, você pode usar uma expressão lambda para implementar uma funcionalidade que também tem suporte por
uma função local. Para obter uma comparação, consulte Funções locais comparadas com expressões Lambda.
Funções locais tornam a intenção do seu código clara. Qualquer pessoa que ler seu código poderá ver que o
método não pode ser chamado, exceto pelo método que o contém. Para projetos de equipe, elas também
impossibilitam que outro desenvolvedor chame o método por engano diretamente de qualquer outro lugar na
classe ou no struct.
NOTE
Antes de C# 8,0, as funções locais não podem incluir o modificador de static . Incluir a palavra-chave static gera o erro
do compilador CS0106, "O modificador 'static' não é válido para este item".
Além disso, os atributos não podem ser aplicados à função local ou aos respectivos parâmetros e parâmetros de
tipo.
O exemplo a seguir define uma função local chamada AppendPathSeparator que é privada para um método
chamado GetText :
using System;
using System.IO;
class Example
{
static void Main()
{
string contents = GetText(@"C:\temp", "example.txt");
Console.WriteLine("Contents of the file:\n" + contents);
}
return filepath;
}
}
}
class Example
{
static void Main()
{
IEnumerable<int> ienum = OddSequence(50, 110);
Console.WriteLine("Retrieved enumerator...");
Em vez disso, você pode lançar uma exceção ao executar a validação e antes de recuperar o iterador, retornando o
iterador de uma função local, conforme mostrado no exemplo a seguir.
using System;
using System.Collections.Generic;
class Example
{
static void Main()
{
IEnumerable<int> ienum = OddSequence(50, 110);
Console.WriteLine("Retrieved enumerator...");
return GetOddSequenceEnumerator();
IEnumerable<int> GetOddSequenceEnumerator()
{
for (int i = start; i <= end; i++)
{
if (i % 2 == 1)
yield return i;
}
}
}
}
// The example displays the following output:
// Unhandled Exception: System.ArgumentOutOfRangeException: Specified argument was out of the range of valid
values.
// Parameter name: end must be less than or equal to 100.
// at Sequence.<GetNumericRange>d__1.MoveNext() in Program.cs:line 23
// at Example.Main() in Program.cs:line 43
Funções locais podem ser usadas de maneira semelhante para lidar com exceções fora da operação assíncrona. Em
geral, exceções geradas no método assíncrono exigem que você examine as exceções internas de um
AggregateException. Funções locais permitem que seu código falhe no modo rápido e permitem que a exceção
seja gerada e observada de forma síncrona.
O exemplo a seguir usa um método assíncrono chamado GetMultipleAsync para pausar para um número
especificado de segundos e retornar um valor que é um múltiplo aleatório desse número de segundos. O atraso
máximo é de 5 segundos; um ArgumentOutOfRangeException resulta em se o valor for maior que 5. Como
mostra o exemplo a seguir, a exceção que é lançada quando um valor de 6 é passada para o método
GetMultipleAsync é encapsulada em um AggregateException depois que o método GetMultipleAsync inicia sua
execução.
using System;
using System.Threading.Tasks;
class Example
{
static void Main()
{
int result = GetMultipleAsync(6).Result;
Console.WriteLine($"The returned value is {result:N0}");
}
Assim como foi feito com o método iterador, podemos refatorar o código deste exemplo para executar a validação
antes de chamar o método assíncrono. Assim como demonstrado pela saída de exemplo a seguir, o
ArgumentOutOfRangeException não é empacotado em uma AggregateException.
using System;
using System.Threading.Tasks;
class Example
{
static void Main()
{
int result = GetMultiple(6).Result;
Console.WriteLine($"The returned value is {result:N0}");
}
return GetValueAsync();
Consulte também
Métodos
Ref returns e ref locals
08/11/2019 • 12 minutes to read • Edit Online
Começando com C# 7.0, C# dá suporte a valores de referência de devolução. Um valor retornado por referência
permite que um método retorne uma referência a uma variável, em vez de um valor, de volta para um chamador.
O chamador pode optar por tratar a variável retornada como se tivesse sido retornada por valor ou referência. O
chamador pode criar uma nova variável que seja uma referência ao valor retornado, chamado de ref local.
Ref locals
Suponha que o método GetContactInformation seja declarado como uma referência de retorno:
Uma atribuição por valor lê o valor de uma variável e o atribui a uma nova variável:
A atribuição anterior declara p como uma variável local. O valor inicial é copiado da leitura do valor retornado
por GetContactInformation . As atribuições futuras para p não alterarão o valor da variável retornado por
GetContactInformation . A variável p não é um alias para a variável retornada.
Você declara uma variável ref local para copiar o alias para o valor original. Na atribuição de seguir, p é um alias
para a variável retornada de GetContactInformation .
O uso subsequente de p é o mesmo que usar a variável retornada pelo GetContactInformation porque p é um
alias dessa variável. As alterações em p também alteram a variável retornada de GetContactInformation .
A palavra-chave ref é usada antes da declaração de variável local e antes da chamada de método.
Você pode acessar um valor por referência da mesma maneira. Em alguns casos, acessar um valor por referência
aumenta o desempenho, evitando uma operação de cópia potencialmente dispendiosa. Por exemplo, a instrução
a seguir mostra como é possível definir um valor de local de ref que é usado para fazer referência a um valor.
A palavra-chave ref é usada antes da declaração da variável local e antes do valor, no segundo exemplo. A falha
ao incluir as palavras-chave ref na declaração e na atribuição da variável nos dois exemplos resulta no erro do
compilador CS8172, "Não é possível inicializar uma variável por referência com um valor."
Antes do C# 7.3, variáveis locais de referência não podiam ser reatribuídas para se referir a um armazenamento
diferente depois de serem inicializadas. Essa restrição foi removida. O exemplo a seguir mostra uma
transferência:
Variáveis locais de referência ainda devem ser inicializadas quando são declaradas.
using System;
class NumberStore
{
int[] numbers = { 1, 3, 7, 15, 31, 63, 127, 255, 511, 1023 };
A exemplo a seguir chama o método NumberStore.FindNumber para recuperar o primeiro valor maior ou igual a
16. O chamador então dobra o valor retornado pelo método. A saída do exemplo mostra a alteração refletida no
valor dos elementos de matriz da instância NumberStore .
Sem suporte para valores retornados por referência, essa operação é executada retornando-se o índice do
elemento de matriz, juntamente com o respectivo valor. O chamador pode usar esse índice para modificar o valor
em uma chamada de método separada. No entanto, o chamador também pode modificar o índice para acessar e
possivelmente modificar outros valores de matriz.
A exemplo a seguir mostra como o método FindNumber poderia ser reescrito após o C# 7.3 para usar a
transferência de local de referência:
using System;
class NumberStore
{
int[] numbers = { 1, 3, 7, 15, 31, 63, 127, 255, 511, 1023 };
Essa segunda versão é mais eficiente, com sequências mais longas em cenários em que o número procurado é
mais próximo ao final da matriz.
Consulte também
ref keyword
Gravação de código segura e eficiente
Passando parâmetros (Guia de Programação em C#)
04/11/2019 • 2 minutes to read • Edit Online
No C#, argumentos podem ser passados para parâmetros por valor ou por referência. A passagem por referência
permite que métodos, propriedades, indexadores, operadores, construtores e membros da função alterem o valor
dos parâmetros e façam essa alteração persistir no ambiente de chamada. Para passar um parâmetro por
referência com a intenção de alterar o valor, use a palavra-chave ref ou out . Para passar por referência com a
intenção de evitar a cópia, mas não alterar o valor, use o modificador in . Para simplificar, somente a palavra-
chave ref é usada nos exemplos neste tópico. Para obter mais informações sobre a diferença entre in , ref e
out , consulte in, ref e out.
class Program
{
static void Main(string[] args)
{
int arg;
// Passing by value.
// The value of arg in Main is not changed.
arg = 4;
squareVal(arg);
Console.WriteLine(arg);
// Output: 4
// Passing by reference.
// The value of arg in Main is changed.
arg = 4;
squareRef(ref arg);
Console.WriteLine(arg);
// Output: 16
}
// Passing by reference
static void squareRef(ref int refParameter)
{
refParameter *= refParameter;
}
}
Especificação da Linguagem C#
Para obter mais informações, veja as listas de argumentos na Especificação da linguagem C#. A especificação da
linguagem é a fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Guia de Programação em C#
Métodos
Passando parâmetros de tipo de valor (Guia de
Programação em C#)
23/10/2019 • 5 minutes to read • Edit Online
Uma variável de tipo de valor contém seus dados diretamente, o que não acontece com uma variável de tipo de
referência, que contém uma referência a seus dados. Passar uma variável de tipo de valor para um método por
valor significa passar uma cópia da variável para o método. Quaisquer alterações no parâmetro que ocorrem
dentro do método não afetam os dados originais armazenados na variável de argumento. Se desejar que o
método chamado altere o valor do argumento, é necessário passá-lo por referência, usando a palavra-chave ref
ou out. Você também pode usar a palavra-chave in para passar um parâmetro de valor por referência, para evitar
a cópia e ainda garantir que o valor não seja alterado. Para simplificar, os exemplos a seguir usam ref .
class PassingValByVal
{
static void SquareIt(int x)
// The parameter x is passed by value.
// Changes to x will not affect the original value of x.
{
x *= x;
System.Console.WriteLine("The value inside the method: {0}", x);
}
static void Main()
{
int n = 5;
System.Console.WriteLine("The value before calling the method: {0}", n);
A variável n é um tipo de valor. Ela contém seus dados, o valor 5 . Quando SquareIt é invocado, o conteúdo de
n é copiado para o parâmetro x , que é elevado ao quadrado dentro do método. No entanto, em Main , o valor
de n é o mesmo depois de chamar o método SquareIt como era antes. A alteração que ocorre dentro do
método afeta apenas a variável local x .
class PassingValByRef
{
static void SquareIt(ref int x)
// The parameter x is passed by reference.
// Changes to x will affect the original value of x.
{
x *= x;
System.Console.WriteLine("The value inside the method: {0}", x);
}
static void Main()
{
int n = 5;
System.Console.WriteLine("The value before calling the method: {0}", n);
Neste exemplo, não é o valor de n que é passado; em vez disso, é passada uma referência a n . O parâmetro x
não é um int; é uma referência a um int , nesse caso, uma referência a n . Portanto, quando x é elevado ao
quadrado dentro do método, o que é realmente é elevado ao quadrado é aquilo a que x se refere, n .
Quando você chama o método SwapByRef , use a palavra-chave ref na chamada, conforme mostrado no
exemplo a seguir.
static void Main()
{
int i = 2, j = 3;
System.Console.WriteLine("i = {0} j = {1}" , i, j);
Consulte também
Guia de Programação em C#
Passando parâmetros
Passando parâmetros de tipo de referência
Passando parâmetros de tipo de referência (Guia de
Programação em C#)
04/11/2019 • 6 minutes to read • Edit Online
Uma variável de um tipo de referência não contém seus dados diretamente; ela contém uma referência a seus
dados. Quando você passa um parâmetro de tipo de referência por valor, é possível alterar os dados que
pertencem ao objeto referenciado, como o valor de um membro de classe. No entanto, não é possível alterar o
valor da referência em si. Por exemplo, não é possível usar a mesma referência para alocar memória para um
novo objeto e fazer com que ele persista fora do bloco. Para fazer isso, passe o parâmetro usando a palavra-chave
ref ou out. Para simplificar, os exemplos a seguir usam ref .
class PassingRefByVal
{
static void Change(int[] pArray)
{
pArray[0] = 888; // This change affects the original element.
pArray = new int[5] {-3, -1, -2, -3, -4}; // This change is local.
System.Console.WriteLine("Inside the method, the first element is: {0}", pArray[0]);
}
Change(arr);
System.Console.WriteLine("Inside Main, after calling the method, the first element is: {0}", arr
[0]);
}
}
/* Output:
Inside Main, before calling the method, the first element is: 1
Inside the method, the first element is: -3
Inside Main, after calling the method, the first element is: 888
*/
No exemplo anterior, a matriz, arr , que é um tipo de referência, é passada para o método sem o parâmetro ref .
Nesse caso, uma cópia da referência, que aponta para arr , é passada para o método. A saída mostra que é
possível para o método alterar o conteúdo de um elemento de matriz, nesse caso de 1 para 888 . No entanto,
alocar uma nova parte da memória usando o operador new dentro do método Change faz a variável pArray
referenciar uma nova matriz. Portanto, quaisquer alterações realizadas depois disso não afetarão a matriz
original, arr , criada dentro de Main . Na verdade, duas matrizes são criadas neste exemplo, uma dentro de Main
e outra dentro do método Change .
Passando tipos de referência por referência
O exemplo a seguir é o mesmo que o exemplo anterior, exceto que a palavra-chave ref é adicionada ao
cabeçalho e à chamada do método. Quaisquer alterações que ocorrem no método afetam a variável original no
programa de chamada.
class PassingRefByRef
{
static void Change(ref int[] pArray)
{
// Both of the following changes will affect the original variables:
pArray[0] = 888;
pArray = new int[5] {-3, -1, -2, -3, -4};
System.Console.WriteLine("Inside the method, the first element is: {0}", pArray[0]);
}
Change(ref arr);
System.Console.WriteLine("Inside Main, after calling the method, the first element is: {0}", arr[0]);
}
}
/* Output:
Inside Main, before calling the method, the first element is: 1
Inside the method, the first element is: -3
Inside Main, after calling the method, the first element is: -3
*/
Todas as alterações que ocorrem dentro do método afetam a matriz original em Main . Na verdade, a matriz
original é realocada usando o operador new . Portanto, depois de chamar o método Change , qualquer referência
a arr aponta para a matriz de cinco elementos, criada no método Change .
Neste exemplo, os parâmetros precisam ser passados por referência para afetar as variáveis no programa de
chamada. Se você remover a palavra-chave ref do cabeçalho de método e da chamada de método, nenhuma
alteração ocorrerá no programa de chamada.
Para obter mais informações sobre cadeias de caracteres, consulte cadeia de caracteres.
Consulte também
Guia de Programação em C#
Passando parâmetros
ref
in
out
Tipos de referência
Como saber a diferença entre passar uma struct e
passar uma referência de classe para um método (C#
guia de programação)
25/11/2019 • 4 minutes to read • Edit Online
O exemplo a seguir demonstra como passar um struct para um método difere de passar uma instância de classe
para um método. No exemplo, ambos os argumentos (struct e instância de classe) são passados por valor e
ambos os métodos alteram o valor de um campo do argumento. No entanto, os resultados dos dois métodos não
são os mesmos, pois o que é passado ao passar um struct é diferente do que é passado ao passar uma instância
de uma classe.
Como um struct é um tipo de valor, ao passar um struct por valor a um método, esse método receberá e operará
em uma cópia do argumento do struct. O método não tem acesso ao struct original no método de chamada e,
portanto, não é possível alterá-lo de forma alguma. O método pode alterar somente a cópia.
Uma instância de classe é um tipo de referência e não é um tipo de valor. Quando um tipo de referência é passado
por valor a um método, esse método receberá uma cópia da referência para a instância da classe. Ou seja, o
método chamado recebe uma cópia do endereço da instância e o método de chamada retém o endereço original
da instância. A instância de classe no método de chamada tem um endereço, o parâmetro do método chamado
tem uma cópia do endereço e os dois endereços se referem ao mesmo objeto. Como o parâmetro contém apenas
uma cópia do endereço, o método chamado não pode alterar o endereço da instância de classe no método de
chamada. No entanto, o método chamado pode usar a cópia do endereço para acessar os membros da classe que
o endereço original e a cópia da referência do endereço. Se o método chamado alterar um membro de classe, a
instância da classe original no método de chamada também será alterada.
O resultado do exemplo a seguir ilustra a diferença. O valor do campo willIChange da instância da classe foi
alterado pela chamada ao método ClassTaker , pois o método usa o endereço no parâmetro para localizar o
campo especificado da instância da classe. O campo willIChange do struct no método de chamada não foi
alterado pela chamada ao método StructTaker , pois o valor do argumento é uma cópia do próprio struct e não
uma cópia de seu endereço. StructTaker altera a cópia e a cópia será perdida quando a chamada para
StructTaker for concluída.
Exemplo
class TheClass
{
public string willIChange;
}
struct TheStruct
{
public string willIChange;
}
class TestClassAndStruct
{
static void ClassTaker(TheClass c)
{
c.willIChange = "Changed";
}
ClassTaker(testClass);
StructTaker(testStruct);
Consulte também
Guia de Programação em C#
Classes
Structs
Passando parâmetros
Variáveis locais de tipo implícito (Guia de
Programação em C#)
25/11/2019 • 9 minutes to read • Edit Online
Variáveis locais podem ser declaradas sem fornecer um tipo explícito. A palavra-chave var instrui o
compilador a inferir o tipo da variável da expressão no lado direito da instrução de inicialização. O tipo
inferido pode ser um tipo interno, um tipo anônimo, um tipo definido pelo usuário ou um tipo definido na
biblioteca de classes .NET Framework. Para obter mais informações sobre como inicializar matrizes com
var , consulte Matrizes de tipo implícito.
Os exemplos a seguir mostram várias maneiras em que as variáveis locais podem ser declaradas com var :
// i is compiled as an int
var i = 5;
// s is compiled as a string
var s = "Hello";
// a is compiled as int[]
var a = new[] { 0, 1, 2 };
É importante entender que a palavra-chave var não significa "variante" e não indica que a variável é
vagamente tipada ou de associação tardia. Isso apenas significa que o compilador determina e atribui o tipo
mais apropriado.
A palavra-chave var pode ser usada nos seguintes contextos:
Em variáveis locais (variáveis declaradas no escopo do método) conforme mostrado no exemplo
anterior.
Em uma instrução de inicialização for.
Para obter mais informações, consulte como usar variáveis locais e matrizes de tipo implícito em uma
expressão de consulta.
class ImplicitlyTypedLocals2
{
static void Main()
{
string[] words = { "aPPLE", "BlUeBeRrY", "cHeRry" };
Comentários
As seguintes restrições se aplicam às declarações de variável de tipo implícito:
var pode ser usado apenas quando uma variável local é declarada e inicializada na mesma
instrução, a variável não pode ser inicializada como nula, um grupo de métodos ou uma função
anônima.
var não pode ser usado em campos no escopo da classe.
Variáveis declaradas usando var não podem ser usadas na expressão de inicialização. Em outras
palavras, essa expressão é legal: int i = (i = 20); mas essa expressão produz um erro de tempo
de compilação: var i = (i = 20);
Diversas variáveis de tipo implícito não podem ser inicializadas na mesma instrução.
Se um tipo nomeado var estiver no escopo, a palavra-chave var será resolvida para esse nome de
tipo e não será tratada como parte de uma declaração de variável local de tipo implícito.
Tipagem implícita com a palavra-chave var só pode ser aplicada às variáveis no escopo do método local.
Digitação implícita não está disponível para os campos de classe, uma vez que o compilador C# encontraria
um paradoxo lógico ao processar o código: o compilador precisa saber o tipo do campo, mas não é possível
determinar o tipo até que a expressão de atribuição seja analisada. A expressão não pode ser avaliada sem
saber o tipo. Considere o código a seguir:
bookTitles é um campo de classe dado o tipo var . Como o campo não tem nenhuma expressão para
avaliar, é impossível para o compilador inferir que tipo bookTitles deveria ser. Além disso, também é
insuficiente adicionar uma expressão ao campo (como você faria para uma variável local):
Quando o compilador encontra campos durante a compilação de código, ele registra cada tipo de campo
antes de processar quaisquer expressões associadas. O compilador encontra o mesmo paradoxo ao tentar
analisar bookTitles : ele precisa saber o tipo do campo, mas o compilador normalmente determinaria o
tipo de var analisando a expressão, o que não possível sem saber o tipo com antecedência.
Você pode descobrir que var também pode ser útil com expressões de consulta em que o tipo construído
exato da variável de consulta é difícil de ser determinado. Isso pode ocorrer com operações de
agrupamento e classificação.
A palavra-chave var também pode ser útil quando o tipo específico da variável é enfadonho de digitar no
teclado, é óbvio ou não acrescenta à legibilidade do código. Um exemplo em que var é útil dessa maneira
é com os tipos genéricos aninhados, como os usados com operações de grupo. Na consulta a seguir, o tipo
da variável de consulta é IEnumerable<IGrouping<string, Student>> . Contanto que você e as outras pessoas
que devem manter o código entendam isso, não há problema em usar a tipagem implícita por questões de
conveniência e brevidade.
// Same as previous example except we use the entire last name as a key.
// Query variable is an IEnumerable<IGrouping<string, Student>>
var studentQuery3 =
from student in students
group student by student.Last;
No entanto, o uso de var tem pelo menos o potencial de tornar seu código mais difícil de entender para
outros desenvolvedores. Por esse motivo, a documentação do C# geralmente usa var somente quando é
necessário.
Consulte também
Referência de C#
Matrizes de tipo implícito
Como usar variáveis locais e matrizes de tipo implícito em uma expressão de consulta
Tipos Anônimos
Inicializadores de objeto e coleção
var
LINQ em C#
LINQ (Consulta Integrada à Linguagem)
for
foreach, in
Instrução using
Como usar variáveis locais e matrizes de tipo
implícito em uma expressão de consultaC# (guia de
programação)
25/11/2019 • 3 minutes to read • Edit Online
Será possível usar variáveis locais de tipo implícito sempre que você desejar que o compilador determine o tipo
de uma variável local. É necessário usar variáveis locais de tipo implícito para armazenar tipos anônimos, usados
frequentemente em expressões de consulta. Os exemplos a seguir ilustram usos obrigatórios e opcionais de
variáveis locais de tipo implícito em consultas.
As variáveis locais de tipo implícito são declaradas usando a palavra-chave contextual var. Para obter mais
informações, consulte Variáveis locais de tipo implícito e Matrizes de tipo implícito.
Exemplo
O exemplo a seguir mostra um cenário comum em que a palavra-chave var é necessária: uma expressão de
consulta que produz uma sequência de tipos anônimos. Nesse cenário, a variável de consulta e a variável de
iteração na instrução foreach devem ser tipadas implicitamente usando var , porque você não tem acesso a um
nome de tipo para o tipo anônimo. Para obter mais informações sobre tipos anônimos, consulte Tipos anônimos.
Exemplo
O exemplo a seguir usa a palavra-chave var em uma situação semelhante, mas na qual o uso de var é opcional.
Como student.LastName é uma cadeia de caracteres, a execução da consulta retorna uma sequência de cadeias de
caracteres. Portanto, o tipo de queryID poderia ser declarado como
System.Collections.Generic.IEnumerable<string> em vez de var . A palavra-chave var é usada por conveniência.
No exemplo, a variável de iteração na instrução foreach tem tipo explícito como uma cadeia de caracteres, mas,
em vez disso, poderia ser declarada usando var . Como o tipo da variável de iteração não é um tipo anônimo, o
uso de var é opcional, não obrigatório. Lembre-se de que var por si só não é um tipo, mas uma instrução para
o compilador inferir e atribuir o tipo.
// Variable queryID could be declared by using
// System.Collections.Generic.IEnumerable<string>
// instead of var.
var queryID =
from student in students
where student.ID > 111
select student.LastName;
Consulte também
Guia de Programação em C#
Métodos de Extensão
LINQ (Consulta Integrada à Linguagem)
var
LINQ em C#
Métodos de extensão (Guia de Programação em
C#)
25/11/2019 • 12 minutes to read • Edit Online
Os métodos de extensão permitem que você "adicione" tipos existentes sem criar um novo tipo derivado,
recompilar ou, caso contrário, modificar o tipo original. Os métodos de extensão são um tipo especial de
método estático, mas são chamados como se fossem métodos de instância no tipo estendido. No caso do
código cliente gravado em C#, F# e Visual Basic, não há nenhuma diferença aparente entre chamar um
método de extensão e os métodos realmente definidos em um tipo.
Os métodos de extensão mais comuns são os operadores de consulta padrão LINQ que adicionam
funcionalidade de consulta aos tipos System.Collections.IEnumerable e
System.Collections.Generic.IEnumerable<T> existentes. Para usar os operadores de consulta padrão, traga-
os primeiro ao escopo com uma diretiva using System.Linq . Em seguida, qualquer tipo que implemente
IEnumerable<T> parece ter métodos de instância como GroupBy, OrderBy, Average e assim por diante.
Você pode exibir esses métodos adicionais no preenchimento de declaração do IntelliSense ao digitar
"ponto" após uma instância de um tipo IEnumerable<T> como List<T> ou Array.
O exemplo a seguir mostra como chamar o método de consulta padrão OrderBy em qualquer matriz de
inteiros. A expressão entre parênteses é uma expressão lambda. Vários operadores de consulta padrão
obtêm expressões lambda como parâmetros, mas isso não é um requisito para métodos de extensão. Para
obter mais informações, consulte Expressões Lambda.
class ExtensionMethods2
{
Os métodos de extensão são definidos como estáticos, mas são chamados usando a sintaxe do método de
instância. Seu primeiro parâmetro especifica em que tipo o método opera e o parâmetro é precedido pelo
modificador this. Os métodos de extensão só estarão no escopo quando você importar explicitamente o
namespace para seu código-fonte com uma diretiva using .
O exemplo a seguir mostra um método de extensão definido para a classe System.String. Observe que isso
é definido em uma classe estática não aninhada e não genérica:
namespace ExtensionMethods
{
public static class MyExtensions
{
public static int WordCount(this String str)
{
return str.Split(new char[] { ' ', '.', '?' },
StringSplitOptions.RemoveEmptyEntries).Length;
}
}
}
O método de extensão WordCount pode ser colocado no escopo com esta diretiva using :
using ExtensionMethods;
Em seu código, você chama o método de extensão com sintaxe de método de instância. No entanto, a
linguagem intermediária (IL ) gerada pelo compilador converte seu código em uma chamada no método
estático. Portanto, o princípio de encapsulamento ainda não está realmente sendo violado. De fato, os
métodos de extensão não podem acessar variáveis particulares no tipo que estão estendendo.
Para obter mais informações, consulte como implementar e chamar um método de extensão personalizado.
Em geral, provavelmente você chamará métodos de extensão com muito mais frequência do que
implementará os seus próprios. Como os métodos de extensão são chamados com a sintaxe do método de
instância, nenhum conhecimento especial é necessário para usá-los no código do cliente. Para habilitar
métodos de extensão para um tipo específico, apenas adicione uma diretiva using para o namespace no
qual os métodos estão definidos. Por exemplo, para usar os operadores de consulta padrão, adicione esta
diretiva using ao seu código:
using System.Linq;
(Você também pode precisar adicionar uma referência a System. Core. dll.) Você observará que os
operadores de consulta padrão agora aparecem no IntelliSense como métodos adicionais disponíveis para
a maioria dos tipos de IEnumerable<T>.
// Define three classes that implement IMyInterface, and then use them to test
// the extension methods.
namespace ExtensionMethodsDemo1
{
using System;
using System;
using Extensions;
using DefineIMyInterface;
class A : IMyInterface
{
public void MethodB() { Console.WriteLine("A.MethodB()"); }
}
class B : IMyInterface
{
public void MethodB() { Console.WriteLine("B.MethodB()"); }
public void MethodA(int i) { Console.WriteLine("B.MethodA(int i)"); }
}
class C : IMyInterface
{
public void MethodB() { Console.WriteLine("C.MethodB()"); }
public void MethodA(object obj)
{
Console.WriteLine("C.MethodA(object obj)");
}
}
class ExtMethodDemo
{
static void Main(string[] args)
{
// Declare an instance of class A, class B, and class C.
A a = new A();
B b = new B();
C c = new C();
Diretrizes gerais
Em geral, recomendamos que você implemente métodos de extensão com parcimônia e somente quando
for necessário. Sempre que possível, o código do cliente que deve estender um tipo existente deve fazer
isso ao criar um novo tipo derivado do tipo existente. Para obter mais informações, consulte Herança.
Ao usar um método de extensão para estender um tipo cujo código-fonte você não pode alterar, há o risco
de uma alteração na implementação do tipo fazer com que o método de extensão interrompa.
Se você implementar métodos de extensão para um determinado tipo, lembre-se das seguintes
considerações:
Um método de extensão nunca será chamado se possuir a mesma assinatura que um método
definido no tipo.
Os métodos de extensão são trazidos para o escopo no nível do namespace. Por exemplo, se você
tiver várias classes estáticas que contenham métodos de extensão em um único namespace
chamado Extensions , todos eles serão trazidos para o escopo pela diretiva using Extensions; .
Para uma biblioteca de classes que você implemente, não use métodos de extensão para evitar incrementar
o número de versão de um assembly. Se desejar adicionar funcionalidade significativa a uma biblioteca da
qual você possua o código-fonte, siga as diretrizes padrão do .NET Framework para controle de versão do
assembly. Para obter mais informações, consulte Controle de versão do assembly.
Consulte também
Guia de Programação em C#
Exemplos de programação paralela (incluem vários métodos de extensão de exemplo)
Expressões Lambda
Visão Geral de Operadores de Consulta Padrão
Regras de conversão para parâmetros de instância e seu impacto
Interoperabilidade de métodos de extensão entre linguagens
Métodos de extensão e representantes via currying
Associação do método de extensão e relatório de erros
Como implementar e chamar um método de
extensão personalizado (C# guia de programação)
25/11/2019 • 3 minutes to read • Edit Online
Este tópico mostra como implementar seus próprios métodos de extensão para qualquer tipo do .NET. O código
de cliente pode usar seus métodos de extensão, adicionando uma referência à DLL que os contém e adicionando
uma diretiva using que especifica o namespace no qual os métodos de extensão são definidos.
Exemplo
O exemplo a seguir implementa um método de extensão chamado WordCount na classe
CustomExtensions.StringExtension . O método funciona na classe String, que é especificada como o primeiro
parâmetro do método. O namespace CustomExtensions é importado para o namespace do aplicativo e o método é
chamado dentro do método Main .
using System.Linq;
using System.Text;
using System;
namespace CustomExtensions
{
// Extension methods must be defined in a static class.
public static class StringExtension
{
// This is the extension method.
// The first parameter takes the "this" modifier
// and specifies the type for which the method is defined.
public static int WordCount(this String str)
{
return str.Split(new char[] {' ', '.','?'}, StringSplitOptions.RemoveEmptyEntries).Length;
}
}
}
namespace Extension_Methods_Simple
{
// Import the extension method namespace.
using CustomExtensions;
class Program
{
static void Main(string[] args)
{
string s = "The quick brown fox jumped over the lazy dog.";
// Call the method as if it were an
// instance method on the type. Note that the first
// parameter is not specified by the calling code.
int i = s.WordCount();
System.Console.WriteLine("Word count of s is {0}", i);
}
}
}
Consulte também
Guia de Programação em C#
Métodos de Extensão
LINQ (Consulta Integrada à Linguagem)
Classes static e membros de classes static
protected
internal
public
this
namespace
Como criar um novo método para uma enumeração
(C# guia de programação)
25/11/2019 • 2 minutes to read • Edit Online
Você pode usar métodos de extensão para adicionar funcionalidades específica para um tipo de enumeração
específico.
Exemplo
No exemplo a seguir, a enumeração Grades representa as letras possíveis que um aluno pode receber em uma
classe. Um método de extensão chamado Passing é adicionado ao tipo Grades de forma que cada instância
desse tipo agora "sabe" se ele representa uma nota de aprovação ou não.
using System;
using System.Collections.Generic;
using System.Text;
using System.Linq;
namespace EnumExtension
{
// Define an extension method in a non-nested static class.
public static class Extensions
{
public static Grades minPassing = Grades.D;
public static bool Passing(this Grades grade)
{
return grade >= minPassing;
}
}
Extensions.minPassing = Grades.C;
Console.WriteLine("\r\nRaising the bar!\r\n");
Console.WriteLine("First {0} a passing grade.", g1.Passing() ? "is" : "is not");
Console.WriteLine("Second {0} a passing grade.", g2.Passing() ? "is" : "is not");
}
}
}
/* Output:
First is a passing grade.
Second is not a passing grade.
Consulte também
Guia de Programação em C#
Métodos de Extensão
Argumentos nomeados e opcionais (Guia de
Programação em C#)
25/11/2019 • 13 minutes to read • Edit Online
Argumentos nomeados
Os argumentos nomeados liberam você da necessidade de lembrar ou procurar a ordem dos parâmetros nas
listas de parâmetros de métodos chamados. O parâmetro para cada argumento pode ser especificado pelo
nome do parâmetro. Por exemplo, uma função que imprime detalhes de pedidos (como o nome do vendedor,
nome do produto e número do pedido) pode ser chamada da maneira padrão, por meio do envio de
argumentos por posição, na ordem definida pela função.
PrintOrderDetails("Gift Shop", 31, "Red Mug");
Se não se lembrar da ordem dos parâmetros, mas souber os nomes, você poderá enviar os argumentos em
qualquer ordem.
PrintOrderDetails(orderNum: 31, productName: "Red Mug", sellerName: "Gift Shop");
Os argumentos nomeados também melhoram a legibilidade do código identificando o que cada argumento
representa. No método de exemplo abaixo, o sellerName não pode ser nulo ou um espaço em branco. Como
sellerName e productName são tipos de cadeia de caracteres, em vez de enviar argumentos por posição, é
melhor usar argumentos nomeados para remover a ambiguidade dos dois e reduzir a confusão para qualquer
pessoa que leia o código.
Os argumentos nomeados, quando usados com argumentos posicionais, são válidos, desde que
não sejam seguidos por argumentos posicionais ou,
PrintOrderDetails("Gift Shop", 31, productName: "Red Mug");
começando com o C# 7.2, sejam usados na posição correta. No exemplo a seguir, o parâmetro orderNum está
na posição correta, mas não está explicitamente nomeado.
PrintOrderDetails(sellerName: "Gift Shop", 31, productName: "Red Mug");
Argumentos posicionais que seguem os argumentos nomeados fora de ordem são inválidos.
// This generates CS1738: Named argument specifications must appear after all fixed arguments have been
specified.
PrintOrderDetails(productName: "Red Mug", 31, "Gift Shop");
Exemplo
O código a seguir implementa os exemplos desta seção, juntamente com outros exemplos.
class NamedExample
{
static void Main(string[] args)
{
// The method can be called in the normal way, by using positional arguments.
PrintOrderDetails("Gift Shop", 31, "Red Mug");
Argumentos opcionais
A definição de um método, construtor, indexador ou delegado pode especificar que seus parâmetros são
obrigatórios ou que são opcionais. Qualquer chamada deve fornecer argumentos para todos os parâmetros
necessários, mas pode omitir argumentos para parâmetros opcionais.
Cada parâmetro opcional tem um valor padrão como parte de sua definição. Se nenhum argumento é enviado
para esse parâmetro, o valor padrão é usado. Um valor padrão deve ser um dos seguintes tipos de expressões:
uma expressão de constante;
uma expressão da forma new ValType() , em que ValType é um tipo de valor, como um enum ou um
struct;
uma expressão da forma default(ValType), em que ValType é um tipo de valor.
Os parâmetros opcionais são definidos no final da lista de parâmetros, depois de todos os parâmetros
obrigatórios. Se o chamador fornecer um argumento para qualquer um de uma sucessão de parâmetros
opcionais, ele deverá fornecer argumentos para todos os parâmetros opcionais anteriores. Não há suporte para
intervalos separados por vírgula na lista de argumentos. Por exemplo, no código a seguir, método de instância
ExampleMethod está definido com um parâmetro obrigatório e dois opcionais.
A chamada para ExampleMethod a seguir causa um erro do compilador, porque um argumento é fornecido para
o terceiro parâmetro, mas não para o segundo.
//anExample.ExampleMethod(3, ,4);
No entanto, se você souber o nome do terceiro parâmetro, poderá usar um argumento nomeado para realizar a
tarefa.
anExample.ExampleMethod(3, optionalint: 4);
O IntelliSense usa colchetes para indicar parâmetros opcionais, conforme mostrado na seguinte ilustração:
NOTE
Você também pode declarar parâmetros opcionais usando a classe OptionalAttribute do .NET. Os parâmetros
OptionalAttribute não exigem um valor padrão.
Exemplo
No exemplo a seguir, o construtor para ExampleClass tem um parâmetro, que é opcional. O método de instância
ExampleMethod tem um parâmetro obrigatório, required e dois parâmetros opcionais, optionalstr e
optionalint . O código em Main mostra as diferentes maneiras como o construtor e o método podem ser
invocados.
namespace OptionalNamespace
{
class OptionalExample
{
static void Main(string[] args)
{
// Instance anExample does not send an argument for the constructor's
// optional parameter.
ExampleClass anExample = new ExampleClass();
anExample.ExampleMethod(1, "One", 1);
anExample.ExampleMethod(2, "Two");
anExample.ExampleMethod(3);
class ExampleClass
{
private string _name;
Interfaces COM
Os argumentos nomeados e opcionais, juntamente com suporte para objetos dinâmicos e outros
aprimoramentos, aprimoram enormemente a interoperabilidade com APIs COM, como APIs de Automação do
Office.
Por exemplo, o método AutoFormat na interface Range do Microsoft Office Excel tem sete parâmetros, todos
opcionais. Esses parâmetros são mostrados na seguinte ilustração:
No C# 3.0 e versões anteriores, é necessário um argumento para cada parâmetro, como mostrado no exemplo a
seguir.
// In C# 3.0 and earlier versions, you need to supply an argument for
// every parameter. The following call specifies a value for the first
// parameter, and sends a placeholder value for the other six. The
// default values are used for those parameters.
var excelApp = new Microsoft.Office.Interop.Excel.Application();
excelApp.Workbooks.Add();
excelApp.Visible = true;
var myFormat =
Microsoft.Office.Interop.Excel.XlRangeAutoFormat.xlRangeAutoFormatAccounting1;
No entanto, você pode simplificar muito a chamada para AutoFormat usando argumentos nomeados e
opcionais, introduzidos no C# 4.0. Os argumentos nomeados e opcionais permitem que você omita o
argumento para um parâmetro opcional se não desejar alterar o valor padrão do parâmetro. Na chamada a
seguir, um valor é especificado para apenas um dos sete parâmetros.
// The following code shows the same call to AutoFormat in C# 4.0. Only
// the argument for which you want to provide a specific value is listed.
excelApp.Range["A1", "B4"].AutoFormat( Format: myFormat );
Para obter mais informações e exemplos, consulte como usar argumentos nomeados e opcionais na
programação do Office e como acessar objetos de interoperabilidade C# do Office usando recursos visuais.
Resolução de sobrecarga
O uso de argumentos nomeados e opcionais afeta a resolução de sobrecarga das seguintes maneiras:
Um método, indexador ou construtor é um candidato para a execução se cada um dos parâmetros é
opcional ou corresponde, por nome ou posição, a um único argumento na instrução de chamada e esse
argumento pode ser convertido para o tipo do parâmetro.
Se mais de um candidato for encontrado, as regras de resolução de sobrecarga de conversões
preferenciais serão aplicadas aos argumentos que são especificados explicitamente. Os argumentos
omitidos para parâmetros opcionais são ignorados.
Se dois candidatos são considerados igualmente bons, a preferência vai para um candidato que não tem
parâmetros opcionais para os quais argumentos foram omitidos na chamada. Esta é uma consequência
da preferência geral na resolução de sobrecarga de candidatos que têm menos parâmetros.
Especificação da Linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a fonte
definitiva para a sintaxe e o uso de C#.
Consulte também
Como: usar argumentos nomeados e opcionais na programação do Office
Usando o tipo dynamic
Usando construtores
Usando indexadores
Como usar argumentos nomeados e opcionais na
programação doC# Office (guia de programação)
25/11/2019 • 8 minutes to read • Edit Online
NOTE
Seu computador pode mostrar diferentes nomes ou locais para alguns dos elementos de interface do usuário do Visual
Studio nas instruções a seguir. A edição do Visual Studio que você possui e as configurações que você usa determinam esses
elementos. Para obter mais informações, consulte Personalizando o IDE.
2. Adicione o seguinte código ao final do método para definir onde exibir texto no documento e qual texto
deve ser exibido:
DisplayInWord();
2. Pressione CTRL+F5 para executar o projeto. É exibido um documento do Word contendo o texto
especificado.
// Convert to a simple table. The table will have a single row with
// three columns.
range.ConvertToTable(Separator: ",");
Em versões anteriores do C#, a chamada para ConvertToTable requer um argumento de referência para
cada parâmetro, conforme mostrado no código a seguir:
2. Para especificar um formato predefinido para a tabela, substitua a última linha em DisplayInWord pela
instrução a seguir e digite CTRL+F5. O formato pode ser qualquer uma das constantes WdTableFormat.
Exemplo
O código a seguir inclui o exemplo completo:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using Word = Microsoft.Office.Interop.Word;
namespace OfficeHowTo
{
class WordProgram
{
static void Main(string[] args)
{
DisplayInWord();
}
// Next, use the ConvertToTable method to put the text into a table.
// The method has 16 optional parameters. You only have to specify
// values for those you want to change.
// Convert to a simple table. The table will have a single row with
// three columns.
range.ConvertToTable(Separator: ",");
Consulte também
Argumentos nomeados e opcionais
Construtores (Guia de Programação em C#)
25/11/2019 • 4 minutes to read • Edit Online
Sempre que uma classe ou struct é criada, o construtor é chamado. Uma classe ou struct pode ter vários
construtores que usam argumentos diferentes. Os construtores permitem que o programador defina valores
padrão, limite a instanciação e grave códigos flexíveis e fáceis de ler. Para obter mais informações e
exemplos, consulte Usando Construtores e Construtores de Instância.
Sintaxe do construtor
Um construtor é um método cujo nome é igual ao nome de seu tipo. Sua assinatura do método inclui
apenas o nome do método e lista de parâmetros, ele não inclui um tipo de retorno. O exemplo a seguir
mostra o construtor para uma classe denominada Person .
Se um construtor puder ser implementado como uma única instrução, você poderá usar uma definição de
corpo da expressão. O exemplo a seguir define uma classe Location cujo construtor tem um único
parâmetro de cadeia de caracteres chamado nome. A definição de corpo da expressão atribui o argumento
ao campo locationName .
public class Location
{
private string locationName;
Construtores estáticos
Os exemplos anteriores têm todos os construtores de instância mostrado, que criam um novo objeto. Uma
classe ou struct também pode ter um construtor estático, que inicializa membros estáticos do tipo.
Construtores estáticos não têm parâmetros. Se você não fornecer um construtor estático para inicializar
campos estáticos, o compilador C# inicializará campos estáticos com seu valor padrão, conforme listado na
Tabela de Valores Padrão.
O exemplo a seguir usa um construtor estático para inicializar um campo estático.
static Adult()
{
minimumAge = 18;
}
Você também pode definir um construtor estático com uma definição de corpo da expressão, como mostra o
exemplo a seguir.
Nesta seção
Usando construtores
Construtores de instância
Construtores particulares
Construtores estáticos
Como escrever um construtor de cópia
Consulte também
Guia de Programação em C#
Classes e Structs
Finalizadores
static
Por que inicializadores são executados na ordem oposta como construtores? Parte um
Usando construtores (Guia de Programação em C#)
04/11/2019 • 7 minutes to read • Edit Online
Quando uma classe ou struct é criado, seu construtor é chamado. Os construtores têm o mesmo nome que a
classe ou struct e eles geralmente inicializam os membros de dados do novo objeto.
No exemplo a seguir, uma classe chamada Taxi é definida usando um construtor simples. A classe é então
instanciada com o operador new. O construtor Taxi é invocado pelo operador new imediatamente após a
memória ser alocada para o novo objeto.
class TestTaxi
{
static void Main()
{
Taxi t = new Taxi();
Console.WriteLine(t.IsInitialized);
}
}
Um construtor que não tem nenhum parâmetro é chamado de construtor sem parâmetros. Os construtores sem
parâmetros são invocados sempre que um objeto é instanciado usando o operador new e nenhum argumento é
fornecido para new . Para obter mais informações, consulte Construtores de instâncias.
A menos que a classe seja static, as classes sem construtores recebem um construtor sem parâmetros público
pelo compilador C# para habilitar a instanciação de classe. Para obter mais informações, consulte Classes
Estáticas e Membros de Classes Estáticas.
Você pode impedir que uma classe seja instanciada tornando o construtor privado, da seguinte maneira:
class NLog
{
// Private Constructor:
private NLog() { }
O código a seguir, no entanto, causa um erro do compilador porque ele não usa new e porque ele tenta usar um
objeto que não foi inicializado:
int i;
Console.WriteLine(i);
Como alternativa, os objetos com base em structs (incluindo todos os tipos numéricos internos) podem ser
inicializados ou atribuídos e, em seguida, usados como no exemplo a seguir:
Portanto, não é necessário chamar o construtor sem parâmetros para um tipo de valor.
Ambas as classes e structs podem definir construtores que usam parâmetros. Os construtores que usam
parâmetros devem ser chamados por meio de uma instrução new ou uma instrução base. As classes e structs
também podem definir vários construtores e nenhum deles precisa definir um construtor sem parâmetros. Por
exemplo:
Essa classe pode ser criada usando qualquer uma das instruções a seguir:
Um construtor pode usar a palavra-chave base para chamar o construtor de uma classe base. Por exemplo:
Neste exemplo, o construtor da classe base é chamado antes de o bloco do construtor ser executado. A palavra-
chave base pode ser usada com ou sem parâmetros. Os parâmetros para o construtor podem ser usados como
parâmetros para base ou como parte de uma expressão. Para obter mais informações, consulte base.
Em uma classe derivada, se um construtor de classe base não for chamado explicitamente usando a palavra-
chave base , o construtor sem parâmetros, se houver, será chamado implicitamente. Isso significa que as
seguintes declarações de construtor são efetivamente iguais:
Se uma classe base não oferecer um construtor sem parâmetros, a classe derivada deverá fazer uma chamada
explícita para um construtor base usando base .
Um construtor pode invocar outro construtor no mesmo objeto usando a palavra-chave this. Como base , this
pode ser usado com ou sem parâmetros e todos os parâmetros no construtor estão disponíveis como
parâmetros para this ou como parte de uma expressão. Por exemplo, o segundo construtor no exemplo
anterior pode ser reescrito usando this :
O uso da palavra-chave this no exemplo anterior faz com que esse construtor seja chamado:
Os construtores podem ser marcados como public, private, protected, internal, protected internal ou private
protected. Esses modificadores de acesso definem como os usuários da classe podem construir a classe. Para
obter mais informações, consulte Modificadores de acesso.
Um construtor pode ser declarado estático usando a palavra-chave static. Os construtores estáticos são
chamados automaticamente, imediatamente antes de qualquer campo estático ser acessado e geralmente são
usados para inicializar membros da classe estática. Para obter mais informações, consulte Construtores estáticos.
Especificação da Linguagem C#
Para obter mais informações, veja Construtores de instância e Construtores estáticos na Especificação de
Linguagem C#. A especificação da linguagem é a fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Guia de Programação em C#
Classes e Structs
Construtores
Finalizadores
Construtores de instâncias (Guia de Programação
em C#)
23/10/2019 • 6 minutes to read • Edit Online
Os construtores de instância são usados para criar e inicializar quaisquer variáveis de membro de instância
quando você usa a expressão new para criar um objeto de uma classe. Para inicializar uma classe estática ou
variáveis estáticas em uma classe não estática, defina um construtor estático. Para obter mais informações,
consulte Construtores estáticos.
O exemplo a seguir mostra um construtor de instância:
class Coords
{
public int x, y;
// constructor
public Coords()
{
x = 0;
y = 0;
}
}
NOTE
Para maior clareza, essa classe contém campos públicos. O uso de campos públicos não é uma prática de programação
recomendada, porque permite que qualquer acesso de programa não restrito e não verificado a trabalhos internos de um
objeto. Geralmente os membros de dados devem ser privados e devem ser acessados apenas por meio de propriedades e
métodos de classe.
Esse construtor de instância é chamado sempre que um objeto com base na classe Coords é criado. Um
construtor como este, que não usa nenhum argumento, é chamado um construtor sem parâmetros. No entanto,
geralmente é útil fornecer construtores adicionais. Por exemplo, podemos adicionar um construtor à classe
Coords que nos permite especificar os valores iniciais para os membros de dados:
Isso permite que objetos Coords sejam criados com os valores iniciais padrão ou específicos, como este:
Se uma classe não tiver um construtor, um construtor sem parâmetros será gerado automaticamente e os
valores padrão serão usados para inicializar os campos de objeto. Por exemplo, um int é inicializada como 0.
Para obter mais informações sobre valores padrão, consulte Tabela de valores padrão. Portanto, como construtor
sem parâmetros da classe Coords inicializa todos os membros de dados como zero, ele pode ser totalmente
removido sem alterar a maneira como a classe funciona. Um exemplo completo que usa vários construtores é
fornecido no Exemplo 1 posteriormente neste tópico e um exemplo de um construtor gerado automaticamente
é fornecido no Exemplo 2.
Construtores de instância também podem ser usados para chamar os construtores de instância de classes base.
O construtor de classe pode invocar o construtor da classe base por meio do inicializador, da seguinte maneira:
Neste exemplo, a classe Circle passa valores que representam o raio e a altura do construtor fornecido pelo
Shape do qual Circle é derivado. Um exemplo completo que usa Shape e Circle é exibido neste tópico como
Exemplo 3.
Exemplo 1
O exemplo a seguir demonstra uma classe com dois construtores de classe, um sem argumentos e outro com
dois argumentos.
class Coords
{
public int x, y;
// Default constructor.
public Coords()
{
x = 0;
y = 0;
}
class MainClass
{
static void Main()
{
var p1 = new Coords();
var p2 = new Coords(5, 3);
Exemplo 2
Neste exemplo, a classe Person não tem nenhum construtor. Nesse caso, um construtor sem parâmetros é
fornecido automaticamente e os campos são inicializados com seus valores padrão.
public class Person
{
public int age;
public string name;
}
class TestPerson
{
static void Main()
{
var person = new Person();
Observe que o valor padrão de age é 0 e o valor padrão de name é null . Para obter mais informações sobre
valores padrão, consulte Tabela de valores padrão.
Exemplo 3:
O exemplo a seguir demonstra como usar o inicializador de classe base. A classe Circle é derivada da classe
geral Shape e a classe Cylinder é derivada da classe Circle . O construtor em cada classe derivada está
usando seu inicializador de classe base.
abstract class Shape
{
public const double pi = Math.PI;
protected double x, y;
class TestShapes
{
static void Main()
{
double radius = 2.5;
double height = 3.0;
Para obter mais exemplos sobre a invocação de construtores de classe base, consulte virtual, override e base.
Consulte também
Guia de Programação em C#
Classes e Structs
Construtores
Finalizadores
static
Construtores particulares (Guia de Programação em
C#)
04/11/2019 • 2 minutes to read • Edit Online
Um construtor particular é um construtor de instância especial. Normalmente, ele é usado em classes que contêm
apenas membros estáticos. Se uma classe tiver um ou mais construtores particulares e nenhum construtor
público, outras classes (exceto as classes aninhadas) não poderão criar instâncias dessa classe. Por exemplo:
class NLog
{
// Private Constructor:
private NLog() { }
A declaração do construtor vazio impede a geração automática de um construtor sem parâmetro. Observe que, se
você não usar um modificador de acesso com o construtor, ele ainda será privado por padrão. No entanto, o
modificador private geralmente é usado explicitamente para deixar claro que a classe não pode ser instanciada.
Construtores particulares são usados para impedir a criação de instâncias de uma classe quando não há métodos
ou campos de instância, como a classe Math ou quando um método é chamado para obter uma instância de uma
classe. Se todos os métodos na classe forem estáticos, considere deixar toda a classe estática. Para obter mais
informações, consulte Classes Estáticas e Membros de Classes Estáticas.
Exemplo
A seguir, temos um exemplo de uma classe usando um construtor particular.
public class Counter
{
private Counter() { }
public static int currentCount;
public static int IncrementCount()
{
return ++currentCount;
}
}
class TestCounter
{
static void Main()
{
// If you uncomment the following statement, it will generate
// an error because the constructor is inaccessible:
// Counter aCounter = new Counter(); // Error
Counter.currentCount = 100;
Counter.IncrementCount();
Console.WriteLine("New count: {0}", Counter.currentCount);
Observe que se você remover a marca de comentário da seguinte instrução do exemplo, ela gerará um erro
porque o construtor está inacessível devido a seu nível de proteção:
Especificação da Linguagem C#
Para obter mais informações, veja Construtores privados na Especificação da Linguagem C#. A especificação da
linguagem é a fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Guia de Programação em C#
Classes e Structs
Construtores
Finalizadores
private
public
Construtores estáticos (Guia de Programação em
C#)
31/10/2019 • 7 minutes to read • Edit Online
Um construtor estático é usado para inicializar quaisquer dados estáticos ou para executar uma ação específica
que precisa ser executada apenas uma vez. Ele é chamado automaticamente antes que a primeira instância seja
criada ou que quaisquer membros estáticos sejam referenciados.
class SimpleClass
{
// Static variable that must be initialized at run time.
static readonly long baseline;
Comentários
Construtores estáticos têm as seguintes propriedades:
Um construtor estático não usa modificadores de acesso nem tem parâmetros.
Uma classe ou struct só pode ter um construtor estático.
Os construtores estáticos não podem ser herdados ou sobrecarregados.
Um construtor estático não pode ser chamado diretamente e destina-se apenas a ser chamado pela
Common Language Runtime (CLR ). Ele é invocado automaticamente.
O usuário não tem controle sobre quando o construtor estático é executado no programa.
Um construtor estático é chamado automaticamente para inicializar a classe antes que a primeira
instância seja criada ou que quaisquer membros estáticos sejam referenciados. Um construtor estático
será executado antes de um construtor de instância. Observe que um construtor de tipo estático é
chamado quando um método estático atribuído a um evento ou um delegado é invocado, e não quando
ele é atribuído. Se os inicializadores de variável de campo estático estiverem presentes na classe do
construtor estático, eles serão executados na ordem textual na qual eles aparecem na declaração de
classe imediatamente antes da execução do construtor estático.
Se você não fornecer um construtor estático para inicializar campos estáticos, todos os campos estáticos
serão inicializados com seu valor padrão, conforme listado na Tabela de Valores Padrão.
Se um construtor estático gera uma exceção, o tempo de execução não o invocará uma segunda vez e o
tipo permanecerá não inicializado durante o tempo de vida do domínio do aplicativo no qual o programa
está sendo executado. Normalmente, uma exceção TypeInitializationException é lançada quando um
construtor estático não consegue instanciar um tipo ou uma exceção sem tratamento que ocorre em um
construtor estático. Para construtores estáticos implícitos que não são definidos explicitamente no
código-fonte, a solução de problemas pode exigir inspeção do código de linguagem intermediária (IL ).
A presença de um construtor estático impede a adição do atributo do tipo BeforeFieldInit. Isso limita a
otimização do tempo de execução.
Um campo declarado como static readonly só pode ser atribuído como parte de sua declaração ou em
um construtor estático. Quando um construtor estático explícito não for necessário, inicialize os campos
estáticos na declaração, em vez de usar um construtor estático para melhorar a otimização do tempo de
execução.
NOTE
Embora não seja diretamente acessível, a presença de um construtor estático explícito deve ser documentada para auxiliar
na solução de problemas de exceções de inicialização.
Uso
Um uso típico de construtores estáticos é quando a classe está usando um arquivo de log e o construtor
é usado para gravar entradas nesse arquivo.
Construtores estáticos também são úteis ao criar classes wrapper para código não gerenciado quando o
construtor pode chamar o método LoadLibrary .
Os construtores estáticos também são um local conveniente para impor verificações em tempo de
execução no parâmetro de tipo que não pode ser verificado em tempo de compilação por meio de
restrições (restrições de parâmetro de tipo).
Exemplo
Nesse exemplo, a classe Bus tem um construtor estático. Quando a primeira instância do Bus for criada (
bus1 ), o construtor estático será invocado para inicializar a classe. O exemplo de saída verifica se o construtor
estático é executado somente uma vez, mesmo se duas instâncias de Bus forem criadas e se é executado antes
que o construtor da instância seja executado.
// Instance constructor.
public Bus(int routeNum)
{
RouteNumber = routeNum;
Console.WriteLine("Bus #{0} is created.", RouteNumber);
}
// Instance method.
public void Drive()
{
TimeSpan elapsedTime = DateTime.Now - globalStartTime;
class TestBus
{
static void Main()
{
// The creation of this instance activates the static constructor.
Bus bus1 = new Bus(71);
Especificação da linguagem C#
Para saber mais, confira a seção Construtores estáticos da Especificação da linguagem C#.
Consulte também
Guia de Programação em C#
Classes e Structs
Construtores
Classes static e membros de classes static
Finalizadores
Diretrizes de design de construtor
Aviso de segurança - CA2121: Construtores estáticos devem ser particulares
Como escrever um construtor de cópia (C# guia de
programação)
25/11/2019 • 2 minutes to read • Edit Online
O C# não fornece um construtor de cópia para objetos, mas é possível escrever um por conta própria.
Exemplo
No exemplo a seguir, a Person class define um construtor de cópia que usa, como seu argumento, uma instância
de Person . Os valores das propriedades do argumento são atribuídos às propriedades da nova instância de
Person . O código contém um construtor de cópia alternativa que envia as propriedades Name e Age da instância
que você deseja copiar para o construtor de instância da classe.
class Person
{
// Copy constructor.
public Person(Person previousPerson)
{
Name = previousPerson.Name;
Age = previousPerson.Age;
}
// Instance constructor.
public Person(string name, int age)
{
Name = name;
Age = age;
}
class TestPerson
{
static void Main()
{
// Create a Person object by using the instance constructor.
Person person1 = new Person("George", 40);
// Show details to verify that the name and age fields are distinct.
Console.WriteLine(person1.Details());
Console.WriteLine(person2.Details());
Consulte também
ICloneable
Guia de Programação em C#
Classes e Structs
Construtores
Finalizadores
Finalizadores (Guia de Programação em C#)
04/11/2019 • 6 minutes to read • Edit Online
Os finalizadores (que também são chamados de destruidores) são usados para executar qualquer limpeza
final necessária, quando uma instância da classe está sendo coletada pelo coletor de lixo.
Comentários
Os finalizadores não podem ser definidos em structs. Eles são usados somente com classes.
Uma classe pode ter somente um finalizador.
Os finalizadores não podem ser herdados ou sobrecarregados.
Os finalizadores não podem ser chamados. Eles são invocados automaticamente.
Um finalizador não usa modificadores ou não tem parâmetros.
Por exemplo, o seguinte é uma declaração de um finalizador para a classe Car .
class Car
{
~Car() // finalizer
{
// cleanup statements...
}
}
Um finalizador também pode ser implementado como uma definição do corpo da expressão, como mostra o
exemplo a seguir.
using System;
O finalizador chama implicitamente Finalize na classe base do objeto. Portanto, uma chamada para um
finalizador é convertida implicitamente para o código a seguir:
NOTE
Finalizadores vazios não devem ser usados. Quando uma classe contém um finalizador, uma entrada é criada na fila
Finalize . Quando o finalizador é chamado, o coletor de lixo é invocado para processar a fila. Um finalizador vazio
apenas resulta na perda de desempenho desnecessária.
O programador não tem controle sobre quando o finalizador é chamado porque isso é determinado pelo
coletor de lixo. O coletor de lixo procura objetos que não estão mais sendo usados pelo aplicativo. Se
considerar um objeto qualificado para finalização, ele chamará o finalizador (se houver) e recuperará a
memória usada para armazenar o objeto.
Em aplicativos .NET Framework (mas não em aplicativos .NET Core), os finalizadores também são chamados
quando o programa é encerrado.
É possível forçar a coleta de lixo chamando Collect, mas na maioria das vezes, isso deve ser evitado porque
pode criar problemas de desempenho.
Exemplo
O exemplo a seguir cria três classes que compõem uma cadeia de herança. A classe First é a classe base,
Second é derivado de First e Third é derivado de Second . Todas as três têm finalizadores. Em Main , uma
instância da classe mais derivada é criada. Quando o programa for executado, observe que os finalizadores das
três classes são chamados automaticamente e em ordem, do mais derivado para o menos derivado.
class First
{
~First()
{
System.Diagnostics.Trace.WriteLine("First's destructor is called.");
}
}
class TestDestructors
{
static void Main()
{
Third t = new Third();
}
}
/* Output (to VS Output Window):
Third's destructor is called.
Second's destructor is called.
First's destructor is called.
*/
Especificação da linguagem C#
Para obter mais informações, confira a seção Destruidores na Especificação da linguagem C#.
Consulte também
IDisposable
Guia de Programação em C#
Construtores
Coleta de lixo
Inicializadores de objeto e coleção (Guia de
Programação em C#)
04/11/2019 • 12 minutes to read • Edit Online
O C# permite criar uma instância de um objeto ou uma coleção e executar as atribuições de membro em
uma única instrução.
Inicializadores de objeto
Os inicializadores de objeto permitem atribuir valores a quaisquer campos ou propriedades acessíveis de
um objeto na hora de criação sem que seja necessário invocar um construtor seguido por linhas de
instruções de atribuição. A sintaxe do inicializador de objeto permite especificar argumentos para um
construtor ou omitir os argumentos (e a sintaxe de parênteses). O exemplo a seguir mostra como usar um
inicializador de objeto com um tipo nomeado, Cat , e como invocar o construtor sem parâmetros. Observe
o uso de propriedades autoimplementadas na classe Cat . Para obter mais informações, consulte
Propriedades autoimplementadas.
public Cat()
{
}
A sintaxe dos inicializadores de objetos permite que você crie uma instância, e depois atribui o objeto recém-
criado, com suas propriedades atribuídas, à variável na atribuição.
Começando com o C# 6, os inicializadores de objeto podem definir indexadores, além de atribuir campos e
propriedades. Considere esta classe Matrix básica:
public class Matrix
{
private double[,] storage = new double[3, 3];
[1, 0] = 0.0,
[1, 1] = 1.0,
[1, 2] = 0.0,
[2, 0] = 0.0,
[2, 1] = 0.0,
[2, 2] = 1.0,
};
Nenhum indexador acessível que contenha um setter acessível pode ser usado como uma das expressões no
inicializador de objeto, independentemente do número ou dos tipos de argumentos. Os argumentos de
índice formam o lado esquerdo da atribuição e o valor é o lado direito da expressão. Por exemplo, estes serão
todos válidos se IndexersExample tiver os indexadores apropriados:
Para que o código anterior seja compilado, o tipo IndexersExample precisará ter os seguintes membros:
Os tipos anônimos habilitam a cláusula select em uma expressão de consulta LINQ a transformar objetos
da sequência original em objetos cujo valor e forma podem diferir dos originais. Isso será útil se você
desejar armazenar apenas uma parte das informações de cada objeto em uma sequência. No exemplo a
seguir, suponha que um objeto de produto ( p ) contenha vários campos e métodos e que você esteja apenas
interessado em criar uma sequência de objetos que contenha o nome do produto e o preço unitário.
var productInfos =
from p in products
select new { p.ProductName, p.UnitPrice };
Quando essa consulta for executada, a variável productInfos conterá uma sequência de objetos que podem
ser acessados em uma instrução foreach como mostrado neste exemplo:
foreach(var p in productInfos){...}
Cada objeto no novo tipo anônimo tem duas propriedades públicas que recebem os mesmos nomes que as
propriedades ou os campos no objeto original. Você também poderá renomear um campo quando estiver
criando um tipo anônimo; o exemplo a seguir renomeia o campo UnitPrice como Price .
Inicializadores de coleção
Os inicializadores de coleção permitem especificar um ou mais inicializadores de elemento quando você
inicializa um tipo de coleção que implementa IEnumerable e tem Add com a assinatura apropriada como
um método de instância ou um método de extensão. Os inicializadores de elemento podem ser um valor
simples, uma expressão ou um inicializador de objeto. Ao usar um inicializador de coleção, você não precisa
especificar várias chamadas. O compilador adiciona as chamadas automaticamente.
O exemplo a seguir mostra dois inicializadores de coleção simples:
O inicializador de coleção a seguir usa inicializadores de objeto para inicializar objetos da classe Cat
definida em um exemplo anterior. Observe que os inicializadores de objeto individuais são envolvidos por
chaves e separados por vírgulas.
Você poderá especificar nulo como um elemento em um inicializador de coleção se o método Add da
coleção permitir.
List<Cat> moreCats = new List<Cat>
{
new Cat{ Name = "Furrytail", Age=5 },
new Cat{ Name = "Peaches", Age=4 },
null
};
O exemplo anterior gera o código que chama o Item[TKey] para definir os valores. Antes C# de 6, você
poderia inicializar dicionários e outros contêineres associativos usando a sintaxe a seguir. Observe que, em
vez da sintaxe do indexador, com parênteses e uma atribuição, ele usa um objeto com vários valores:
Este exemplo de inicializador chama Add(TKey, TValue) para adicionar os três itens no dicionário. Essas duas
maneiras diferentes para inicializar coleções associativas tem um comportamento um pouco diferente
devido às chamadas de método que o compilador gera. As duas variantes trabalham com a classe
Dictionary . Outros tipos podem ser compatíveis apenas com uma ou com outra, dependendo da API
pública deles.
Exemplos
O exemplo a seguir combina os conceitos de inicializadores de coleção e objeto.
public class InitializationSample
{
public class Cat
{
// Auto-implemented properties.
public int Age { get; set; }
public string Name { get; set; }
public Cat() { }
// Display results.
System.Console.WriteLine(cat.Name);
O exemplo a seguir mostra um objeto que implementa IEnumerable e contém um método Add com vários
parâmetros. Ele usa um inicializador de coleção com vários elementos por item na lista que correspondem à
assinatura do método Add .
public class FullExample
{
class FormattedAddresses : IEnumerable<string>
{
private List<string> internalList = new List<string>();
public IEnumerator<string> GetEnumerator() => internalList.GetEnumerator();
Console.WriteLine("Address Entries:");
/*
* Prints:
Address Entries:
John Doe
123 Street
Topeka, KS 00000
Jane Smith
456 Street
Topeka, KS 00000
*/
}
Os métodos Add podem usar a palavra-chave params para obter um número variável de argumentos,
como mostrado no seguinte exemplo. Este exemplo também demonstra a implementação personalizada de
um indexador para inicializar uma coleção usando índices.
public void Add(TKey key, params TValue[] values) => Add(key, (IEnumerable<TValue>)values);
storedValues.AddRange(values);
}
}
/*
* Prints:
Using second multi-valued dictionary created with a collection initializer using indexing:
Using third multi-valued dictionary created with a collection initializer using indexing:
Consulte também
Guia de Programação em C#
LINQ em C#
Tipos Anônimos
Como inicializar objetos usando um inicializador de
objeto (C# guia de programação)
25/11/2019 • 3 minutes to read • Edit Online
Você pode usar os inicializadores de objeto para inicializar objetos de tipo de maneira declarativa, sem invocar
explicitamente um construtor para o tipo.
Os exemplos a seguir mostram como usar os inicializadores de objeto com objetos nomeados. O compilador
processa os inicializadores de objetos ao acessar, primeiro, o construtor de instância padrão e, em seguida,
processar as inicializações de membro. Portanto, se o construtor sem parâmetros for declarado como private na
classe, os inicializadores de objeto que exigem acesso público falharão.
Se você estiver definindo um tipo anônimo, é necessário usar um inicializador de objeto. Para obter mais
informações, consulte como retornar subconjuntos de propriedades de elemento em uma consulta.
Exemplo
O exemplo a seguir mostra como inicializar um novo tipo StudentName , usando inicializadores de objeto. Este
exemplo define as propriedades de StudentName tipo:
Console.WriteLine(student1.ToString());
Console.WriteLine(student2.ToString());
Console.WriteLine(student2.ToString());
Console.WriteLine(student3.ToString());
Console.WriteLine(student4.ToString());
}
// Output:
// Craig 0
// Craig 0
// 183
// Craig 116
// Properties.
public string FirstName { get; set; }
public string LastName { get; set; }
public int ID { get; set; }
Os inicializadores de objeto podem ser usados para definir indexadores em um objeto. O exemplo a seguir define
uma classe BaseballTeam que usa um indexador para obter e definir jogadores em posições diferentes. O
inicializador pode atribuir jogadores com base na abreviação da posição ou no número usado para cada scorecard
de beisebol de posição:
public class HowToIndexInitializer
{
public class BaseballTeam
{
private string[] players = new string[9];
private readonly List<string> positionAbbreviations = new List<string>
{
"P", "C", "1B", "2B", "3B", "SS", "LF", "CF", "RF"
};
Console.WriteLine(team["2B"]);
}
}
Consulte também
Guia de Programação em C#
Inicializadores de objeto e coleção
Como inicializar um dicionário com um inicializador
de coleção (Guia de Programação em C#)
23/10/2019 • 2 minutes to read • Edit Online
Um Dictionary<TKey,TValue> contém uma coleção de pares de chave-valor. Seu método Add recebe dois
parâmetros, um para a chave e outro para o valor. Uma maneira de inicializar um Dictionary<TKey,TValue> ou
qualquer coleção cujo método Add use vários parâmetros, é colocar cada conjunto de parâmetros entre chaves,
conforme mostrado no exemplo a seguir. Outra opção é usar um inicializador de índice, também mostrado no
exemplo a seguir.
Exemplo
No exemplo de código a seguir, um Dictionary<TKey,TValue> é inicializado com instâncias do tipo StudentName . A
primeira inicialização usa o método Add com dois argumentos. O compilador gera uma chamada para Add para
cada um dos pares de chaves int e valores StudentName . A segunda usa um método de indexador público de
leitura/gravação da classe Dictionary :
Observe os dois pares de chaves em cada elemento da coleção na primeira declaração. As chaves internas
circundam o inicializador de objeto do StudentName e as chaves mais externas circundam o inicializador do par
chave/valor que será adicionado ao students Dictionary<TKey,TValue>. Por fim, todo o inicializador de coleção
do dicionário é colocado entre chaves. Na segunda inicialização, o lado esquerdo da atribuição é a chave e o lado
direito é o valor, usando um inicializador de objeto para StudentName .
Consulte também
Guia de Programação em C#
Inicializadores de objeto e coleção
Tipos aninhados (Guia de Programação em C#)
23/10/2019 • 2 minutes to read • Edit Online
Um tipo definido em uma classe ou struct é denominado tipo aninhado. Por exemplo:
class Container
{
class Nested
{
Nested() { }
}
}
Independentemente de o tipo externo ser uma classe ou uma estrutura, tipos aninhados tornam-se privados por
padrão; eles são acessíveis somente de seu tipo recipiente. No exemplo anterior, a classe Nested é inacessível
para tipos externos.
Você também pode especificar um modificador de acesso para definir a acessibilidade de um tipo aninhado, da
seguinte maneira:
Os tipos aninhados de uma classe podem ser public, protected, internal, protected internal, private ou
private protected.
No entanto, a definição de uma classe aninhada protected , protected internal ou private protected
dentro de uma classe selada gera um aviso do compilador CS0628, "novo membro protegido declarado na
classe selada".
Tipos aninhados de um struct podem ser públicos, internos ou particulares.
O exemplo a seguir torna a classe Nested pública:
class Container
{
public class Nested
{
Nested() { }
}
}
O tipo aninhado ou interno pode acessar o tipo recipiente ou externo. Para acessar o tipo recipiente, passe-o
como um argumento ao construtor do tipo aninhado. Por exemplo:
public class Container
{
public class Nested
{
private Container parent;
public Nested()
{
}
public Nested(Container parent)
{
this.parent = parent;
}
}
}
Um tipo aninhado tem acesso a todos os membros acessíveis ao seu tipo recipiente. Ele pode acessar membros
privados e protegidos do tipo recipiente, incluindo quaisquer membros protegidos herdados.
Na declaração anterior, o nome completo da classe Nested é Container.Nested . Este é o nome usado para criar
uma nova instância da classe aninhada, da seguinte maneira:
Consulte também
Guia de Programação em C#
Classes e Structs
Modificadores de acesso
Construtores
Classes e métodos partial (Guia de Programação em
C#)
04/11/2019 • 12 minutes to read • Edit Online
É possível dividir a definição de uma classe ou struct, uma interface ou um método em dois ou mais arquivos de
origem. Cada arquivo de origem contém uma seção da definição de tipo ou método e todas as partes são
combinadas quando o aplicativo é compilado.
Classes parciais
Há várias situações em que a divisão de uma definição de classe é desejável:
Ao trabalhar em projetos grandes, dividir uma classe em arquivos separados permite que vários
programadores trabalhem ao mesmo tempo.
Ao trabalhar com código-fonte gerado automaticamente, o código pode ser adicionado à classe sem
precisar recriar o arquivo de origem. O Visual Studio usa essa abordagem quando cria Windows Forms,
código de wrapper de serviço Web e assim por diante. Você pode criar código que usa essas classes sem
precisar modificar o arquivo que o Visual Studio cria.
Para dividir uma definição de classe, use o modificador de palavra-chave partial, como mostrado aqui:
A palavra-chave partial indica que outras partes da classe, struct ou interface podem ser definidas no
namespace. Todas as partes devem usar a palavra-chave partial . Todas as partes devem estar disponíveis em
tempo de compilação para formar o tipo final. Todas as partes devem ter a mesma acessibilidade, tais como
public , private e assim por diante.
Se alguma parte for declarada como abstrata, o tipo inteiro será considerado abstrato. Se alguma parte for
declarada como lacrada, o tipo inteiro será considerado lacrado. Se alguma parte declarar um tipo base, o tipo
inteiro herda dessa classe.
Todas as partes que especificam uma classe base devem concordar, mas partes que omitem uma classe base
ainda herdam o tipo base. As partes podem especificar diferentes interfaces base e o tipo final implementa todas
as interfaces listadas por todas as declarações parciais. Qualquer membro de classe, struct ou interface declarado
em uma definição parcial está disponível para todas as outras partes. O tipo final é a combinação de todas as
partes em tempo de compilação.
NOTE
O modificador partial não está disponível em declarações de enumeração ou delegados.
O exemplo a seguir mostra que tipos aninhados podem ser parciais, mesmo se o tipo no qual eles estão
aninhados não for parcial.
class Container
{
partial class Nested
{
void Test() { }
}
partial class Nested
{
void Test2() { }
}
}
Em tempo de compilação, atributos de definições de tipo parcial são mesclados. Por exemplo, considere as
declarações a seguir:
[SerializableAttribute]
partial class Moon { }
[ObsoleteAttribute]
partial class Moon { }
[SerializableAttribute]
[ObsoleteAttribute]
class Moon { }
Restrições
Há várias regras para seguir quando você está trabalhando com definições de classes parciais:
Todas as definições de tipo parcial que devem ser partes do mesmo tipo devem ser modificadas com
partial . Por exemplo, as seguintes declarações de classe geram um erro:
O modificador partial só pode aparecer imediatamente antes das palavras-chave class , struct ou
interface .
Tipos parciais aninhados são permitidos em definições de tipo parcial, conforme ilustrado no exemplo a
seguir:
Todas as definições de tipo parcial que devem ser partes do mesmo tipo devem ser definidas no mesmo
assembly e no mesmo módulo (arquivo .dll ou .exe). Definições parciais não podem abranger vários
módulos.
O nome de classe e os parâmetros de tipo genérico devem corresponder em todas as definições de tipo
parcial. Tipos genéricos podem ser parciais. Cada declaração parcial deve usar os mesmos nomes de
parâmetro na mesma ordem.
As seguintes palavras-chave em uma definição de tipo parcial são opcionais, mas, se estiverem presentes
em uma definição de tipo parcial, não podem entrar em conflito com as palavras-chave especificadas em
outra definição parcial para o mesmo tipo:
public
private
protected
internal
abstract
sealed
classe base
modificador new (partes aninhadas)
restrições genéricas
Para obter mais informações, consulte Restrições a parâmetros de tipo.
Exemplo 1
Descrição
No exemplo a seguir, os campos e o construtor da classe, Coords , são declarados em uma definição de classe
parcial e o membro, PrintCoords , é declarado em outra definição de classe parcial.
Código
class TestCoords
{
static void Main()
{
Coords myCoords = new Coords(10, 15);
myCoords.PrintCoords();
Exemplo 2
Descrição
O exemplo a seguir mostra que você também pode desenvolver interfaces e structs parciais.
Código
partial interface ITest
{
void Interface_Test();
}
partial struct S1
{
void Struct_Test() { }
}
partial struct S1
{
void Struct_Test2() { }
}
Métodos parciais
Uma classe ou struct parcial pode conter um método parcial. Uma parte da classe contém a assinatura do
método. Uma implementação opcional pode ser definida na mesma parte ou em outra parte. Se a
implementação não for fornecida, o método e todas as chamadas para o método serão removidos em tempo de
compilação.
Métodos parciais permitem que o implementador de uma parte de uma classe defina um método, semelhante a
um evento. O implementador da outra parte da classe pode decidir implementará o método ou não. Se o
método não for implementado, o compilador removerá a assinatura do método e todas as chamadas para o
método. As chamadas para o método, incluindo qualquer resultado que ocorreria da avaliação de argumentos
nas chamadas, não têm efeito em tempo de execução. Portanto, qualquer código na classe parcial pode usar
livremente um método parcial, mesmo que a implementação não seja fornecida. Nenhum erro de tempo de
compilação ou tempo de execução ocorrerá se o método for chamado, mas não implementado.
Métodos parciais são especialmente úteis como uma maneira de personalizar o código gerado. Eles permitem
que um nome e uma assinatura de método sejam reservados, para que o código gerado possa chamar o
método, mas o desenvolvedor possa decidir se deve implementar o método. Assim como as classes parciais, os
métodos parciais habilitam o código criado por um gerador de código e o código criado por um desenvolvedor
humano a funcionarem juntos sem custos de tempo de execução.
Uma declaração de método parcial consiste em duas partes: a definição e a implementação. Elas podem estar em
partes separadas de uma classe parcial ou na mesma parte. Se não houver nenhuma declaração de
implementação, o compilador otimizará a declaração de definição e todas as chamadas para o método.
// Definition in file1.cs
partial void onNameChanged();
// Implementation in file2.cs
partial void onNameChanged()
{
// method body
}
Declarações de métodos parcial devem começar com a palavra-chave contextual partial e o método deve
retornar void.
Os métodos parciais podem ter os parâmetros in ou ref, mas não out.
Métodos parciais são implicitamente private e portanto não podem ser virtual.
Métodos parciais não podem ser extern, pois a presença do corpo determina se eles estão sendo definidos
ou implementados.
Métodos parciais podem ter modificadores static e unsafe.
Métodos parciais podem ser genéricos. Restrições são colocadas quanto à declaração de método parcial
de definição e, opcionalmente, podem ser repetidas na de implementação. Nomes de parâmetro e de tipo
de parâmetro não precisam ser iguais na declaração de implementação e na de definição.
Você pode fazer um delegado para um método parcial que foi definido e implementado, mas não para um
método parcial que só foi definido.
Especificação da Linguagem C#
Para obter mais informações, veja Tipos parciais na Especificação da Linguagem C#. A especificação da
linguagem é a fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Guia de Programação em C#
Classes
Structs
Interfaces
(partial (tipo)
Tipos anônimos (Guia de Programação em C#)
04/11/2019 • 7 minutes to read • Edit Online
// Rest the mouse pointer over v.Amount and v.Message in the following
// statement to verify that their inferred types are int and string.
Console.WriteLine(v.Amount + v.Message);
Os tipos anônimos são normalmente usados na cláusula select de uma expressão de consulta para
retornar um subconjunto das propriedades de cada objeto na sequência de origem. Para obter mais
informações sobre consultas, consulte LINQ C#in .
Os tipos anônimos contêm uma ou mais propriedades públicas somente leitura. Nenhum outro tipo de
membros da classe, como métodos ou eventos, é válido. A expressão que é usada para inicializar uma
propriedade não pode ser null , uma função anônima ou um tipo de ponteiro.
O cenário mais comum é inicializar um tipo anônimo com propriedades de outro tipo. No exemplo a
seguir, suponha que existe uma classe com o nome Product . A classe Product inclui as propriedades
Color e Price , além de outras propriedades que não lhe interessam. A variável products é uma
coleção de objetos do Product . A declaração do tipo anônimo começa com a palavra-chave new . A
declaração inicializa um novo tipo que usa apenas duas propriedades de Product . Isso faz com que uma
menor quantidade de dados seja retornada na consulta.
Quando você não especifica os nomes de membros no tipo anônimo, o compilador dá aos membros de
tipo anônimo o mesmo nome da propriedade que está sendo usada para inicializá-los. Forneça um nome
para a propriedade que está sendo inicializada com uma expressão, como mostrado no exemplo anterior.
No exemplo a seguir, os nomes das propriedades do tipo anônimo são Color e Price .
var productQuery =
from prod in products
select new { prod.Color, prod.Price };
Normalmente, ao usar um tipo anônimo para inicializar uma variável, a variável é declarada como uma
variável local de tipo implícito usando var. O nome do tipo não pode ser especificado na declaração da
variável, porque apenas o compilador tem acesso ao nome subjacente do tipo anônimo. Para obter mais
informações sobre var , consulte Variáveis de local digitadas implicitamente.
Você pode criar uma matriz de elementos de tipo anônimo combinando uma variável local de tipo
implícito e uma matriz de tipo implícito, como mostrado no exemplo a seguir.
var anonArray = new[] { new { name = "apple", diam = 4 }, new { name = "grape", diam = 1 }};
Comentários
Os tipos anônimos são tipos class que derivam diretamente de object e que não podem ser convertidos
para qualquer tipo, exceto object. O compilador fornece um nome para cada tipo anônimo, embora o seu
aplicativo não possa acessá-lo. Do ponto de vista do Common Language Runtime, um tipo anônimo não
é diferente de qualquer outro tipo de referência.
Se dois ou mais inicializadores de objeto anônimos em um assembly especificarem uma sequência de
propriedades que estão na mesma ordem e que têm os mesmos nomes e tipos, o compilador tratará os
objetos como instâncias do mesmo tipo. Eles compartilham o mesmo tipo de informação gerado pelo
compilador.
Você não pode declarar que um campo, uma propriedade, um evento ou um tipo de retorno de um
método tem um tipo anônimo. Da mesma forma, não pode declarar que um parâmetro formal de um
método, propriedade, construtor ou indexador tem um tipo anônimo. Para passar um tipo anônimo ou
uma coleção que contenha tipos anônimos, como um argumento para um método, você pode declarar o
parâmetro como objeto type. No entanto, isso anula a finalidade dos tipos fortes. Se você precisa
armazenar os resultados da consulta ou passá-los fora do limite do método, considere o uso de uma
estrutura ou classe com denominação comum em vez de um tipo anônimo.
Como os métodos Equals e GetHashCode em tipos anônimos são definidos em termos dos métodos das
propriedades Equals e GetHashCode , duas instâncias do mesmo tipo anônimo são iguais somente se
todas as suas propriedades forem iguais.
Consulte também
Guia de Programação em C#
Inicializadores de objeto e coleção
Introdução a LINQ em C#
LINQ em C#
Como retornar subconjuntos de propriedades de
elemento em uma consultaC# (guia de
programação)
25/11/2019 • 2 minutes to read • Edit Online
Use um tipo anônimo em uma expressão de consulta quando essas duas condições se aplicarem:
Você deseja retornar apenas algumas das propriedades de cada elemento de origem.
Você não precisa armazenar os resultados da consulta fora do escopo do método em que a consulta é
executada.
Se você deseja retornar apenas uma propriedade ou campo de cada elemento de origem, use somente o operador
de ponto na cláusula select . Por exemplo, para retornar somente a ID de cada student , escreva a cláusula
select da seguinte maneira:
select student.ID;
Exemplo
O exemplo a seguir mostra como usar um tipo anônimo para retornar apenas um subconjunto das propriedades
de cada elemento de origem que corresponda à condição especificada.
Observe que o tipo anônimo usa nomes do elemento de origem para suas propriedades, se nenhum nome for
especificado. Para fornecer novos nomes para as propriedades no tipo anônimo, escreva a instrução select da
seguinte maneira:
select new { First = student.FirstName, Last = student.LastName };
Se você tentar fazer isso no exemplo anterior, a instrução Console.WriteLine também deve ser alterada:
Compilando o código
Para executar esse código, copie e cole a classe em um aplicativo de console em C# com uma diretiva using para
System.Linq.
Consulte também
Guia de Programação em C#
Tipos Anônimos
LINQ em C#
Interfaces (Guia de Programação em C#)
31/10/2019 • 8 minutes to read • Edit Online
Uma interface contém definições para um grupo de funcionalidades relacionadas que uma classe não
abstrata ou uma struct deve implementar.
Usando interfaces, você pode, por exemplo, incluir o comportamento de várias fontes em uma classe. Essa
funcionalidade é importante em C# porque a linguagem não dá suporte a várias heranças de classes. Além
disso, use uma interface se você deseja simular a herança para structs, pois eles não podem herdar de outro
struct ou classe.
Você define uma interface usando a palavra-chave interface. como mostrado no exemplo a seguir.
interface IEquatable<T>
{
bool Equals(T obj);
}
O nome da struct deve ser um nome do identificador válido em C#. Por convenção, os nomes de interface
começam com uma letra maiúscula I .
Qualquer classe ou struct que implemente a interface IEquatable<T> deve conter uma definição para um
método Equals que corresponda à assinatura que a interface especifica. Como resultado, você pode contar
com uma classe que implementa IEquatable<T> para conter um método Equals com o qual uma instância
da classe pode determinar se é igual a outra instância da mesma classe.
A definição de IEquatable<T> não fornece uma implementação para Equals . Uma classe ou estrutura pode
implementar várias interfaces, mas uma classe só pode herdar de uma única classe.
Para obter mais informações sobre classes abstratas, consulte Classes e membros de classes abstratos e
lacrados.
As interfaces podem conter métodos, propriedades, eventos, indexadores ou qualquer combinação desses
quatro tipos de membro. Para obter links para exemplos, consulte as seções relacionadas. Uma interface não
pode conter constantes, campos, operadores, construtores de instância, finalizadores ou tipos. Os membros
da interface são automaticamente públicos e eles não podem incluir nenhum modificador de acesso. Os
membros também não podem ser estáticos.
Para implementar um membro de interface, o membro correspondente da classe de implementação deve
ser público, não estático e ter o mesmo nome e assinatura do membro de interface.
Quando uma classe ou struct implementa uma interface, a classe ou o struct deve fornecer uma
implementação para todos os membros que a interface define. A interface não fornece nenhuma
funcionalidade que uma classe ou um struct possa herdar da forma que ela pode herdar a funcionalidade da
classe base. No entanto, se uma classe base implementa uma interface, qualquer classe que é derivada da
classe base herda essa implementação.
O exemplo a seguir mostra uma implementação da interface IEquatable<T>. A classe de implementação,
Car , deverá fornecer uma implementação do método Equals.
public class Car : IEquatable<Car>
{
public string Make {get; set;}
public string Model { get; set; }
public string Year { get; set; }
As propriedades e os indexadores de uma classe podem definir acessadores extras para uma propriedade
ou o indexador que é definido em uma interface. Por exemplo, uma interface pode declarar uma
propriedade que tem um acessador get. A classe que implementa a interface pode declarar a mesma
propriedade tanto com um acessador get quanto com um set. No entanto, se a propriedade ou o
indexador usa a implementação explícita, os acessadores devem corresponder. Para obter mais informações
sobre a implementação explícita, consulte Implementação de interface explícita e Propriedades da interface.
As interfaces podem herdar de outras interfaces. Uma classe pode incluir uma interface várias vezes por
meio das classes base que ela herda ou por meio de interfaces que outras interfaces herdam. No entanto, a
classe poderá fornecer uma implementação de uma interface apenas uma vez e somente se a classe declarar
a interface como parte da definição de classe ( class ClassName : InterfaceName ). Se a interface é herdada
porque é herdada de uma classe base que implementa a interface, a classe base fornece a implementação
dos membros da interface. No entanto, a classe derivada pode reimplementar qualquer membro de
interface virtual em vez de usar a implementação herdada.
Uma classe base também pode implementar membros de interface usando membros virtuais. Nesse caso,
uma classe derivada pode alterar o comportamento da interface substituindo os membros virtuais. Para
obter mais informações sobre membros virtuais, consulte Polimorfismo.
Resumo de interfaces
Uma interface tem as propriedades a seguir:
Uma interface é como uma classe base abstrata que contém apenas membros abstratos. Qualquer classe
ou struct que implementa a interface deve implementar todos os seus membros.
Uma interface não pode ser instanciada diretamente. Seus membros são implementados por qualquer
classe ou struct que implemente a interface.
As interfaces podem conter propriedades, indexadores, métodos e eventos.
As interfaces não têm implementações de métodos.
Uma classe ou struct pode implementar várias interfaces. Uma classe pode herdar uma classe base e
também implementar uma ou mais interfaces.
Nesta seção
Implementação de interface explícita
Explica como criar um membro da classe que é específico para uma interface.
Como implementar membros de interface explicitamente
Fornece um exemplo de como implementar explicitamente membros de interfaces.
Como implementar membros de duas interfaces explicitamente
Fornece um exemplo de como implementar explicitamente membros de interfaces com herança.
Seções relacionadas
Propriedades de interface
Indexadores em interfaces
Como implementar eventos de interface
Classes e Structs
Herança
Métodos
Polimorfismo
Classes e membros de classes abstract e sealed
Propriedades
Eventos
Indexadores
Consulte também
Guia de Programação em C#
Herança
Nomes de identificadores
Implementação de interface explícita (Guia de
Programação em C#)
23/10/2019 • 3 minutes to read • Edit Online
Caso uma classe implemente duas interfaces que contêm um membro com a mesma assinatura, logo,
implementar esse membro na classe fará com que as duas interfaces usem tal membro como sua
implementação. No exemplo a seguir, todas as chamadas para Paint invocam o mesmo método.
class Test
{
static void Main()
{
SampleClass sc = new SampleClass();
IControl ctrl = sc;
ISurface srfc = sc;
interface IControl
{
void Paint();
}
interface ISurface
{
void Paint();
}
class SampleClass : IControl, ISurface
{
// Both ISurface.Paint and IControl.Paint call this method.
public void Paint()
{
Console.WriteLine("Paint method in SampleClass");
}
}
// Output:
// Paint method in SampleClass
// Paint method in SampleClass
// Paint method in SampleClass
No entanto, se os dois membros de interface não executarem a mesma função, isso pode levar a uma
implementação incorreta de uma ou ambas as interfaces. É possível implementar um membro de interface
explicitamente — criando um membro de classe que seja chamado apenas por meio da interface e seja
específico para essa interface. Isso é realizado ao nomear o membro da classe com o nome da interface e um
ponto final. Por exemplo:
public class SampleClass : IControl, ISurface
{
void IControl.Paint()
{
System.Console.WriteLine("IControl.Paint");
}
void ISurface.Paint()
{
System.Console.WriteLine("ISurface.Paint");
}
}
O membro da classe IControl.Paint está disponível somente por meio da interface IControl e ISurface.Paint
está disponível apenas por meio do ISurface . Ambas as implementações de método são separadas e nenhuma
delas está diretamente disponível na classe. Por exemplo:
IControl c = obj;
c.Paint(); // Calls IControl.Paint on SampleClass.
ISurface s = obj;
s.Paint(); // Calls ISurface.Paint on SampleClass.
// Output:
// IControl.Paint
// ISurface.Paint
A implementação explícita também é utilizada para resolver casos em que duas interfaces declaram membros
diferentes com o mesmo nome, como uma propriedade e um método:
interface ILeft
{
int P { get;}
}
interface IRight
{
int P();
}
Para implementar as duas interfaces, é necessário que uma classe use a implementação explícita para a
propriedade P, o método P ou ambos, a fim de evitar um erro do compilador. Por exemplo:
Consulte também
Guia de Programação em C#
Classes e Structs
Interfaces
Herança
Como: implementar membros de interface
explicitamente (Guia de Programação em C#)
23/10/2019 • 2 minutes to read • Edit Online
Este exemplo declara uma interface, IDimensions e uma classe, Box , que implementa explicitamente os membros
de interface getLength e getWidth . Os membros são acessados por meio da instância dimensions da interface.
Exemplo
interface IDimensions
{
float getLength();
float getWidth();
}
Programação robusta
Observe que as seguintes linhas, no método Main , foram comentadas, pois produziriam erros de
compilação. Um membro de interface implementado explicitamente não pode ser acessado de uma
instância de classe:
Observe também que as linhas a seguir, no método Main , imprimem com êxito as dimensões da caixa,
pois os métodos estão sendo chamados de uma instância da interface:
System.Console.WriteLine("Length: {0}", dimensions.getLength());
System.Console.WriteLine("Width: {0}", dimensions.getWidth());
Consulte também
Guia de Programação em C#
Classes e Structs
Interfaces
Como: implementar membros de duas interfaces de forma explícita
Como: implementar membros de duas interfaces
explicitamente (Guia de Programação em C#)
23/10/2019 • 2 minutes to read • Edit Online
A implementação explícita da interface também permite ao programador implementar duas interfaces que têm os
mesmos nomes de membro e implementar separadamente cada membro de interface. Este exemplo exibe as
dimensões de uma caixa em unidades inglesas e no sistema métrico. A Caixa classe implementa duas interfaces,
IEnglishDimensions e IMetricDimensions, que representam os diferentes sistemas de medida. As duas interfaces
têm nomes de membro idênticos, Comprimento e Largura.
Exemplo
// Declare the English units interface:
interface IEnglishDimensions
{
float Length();
float Width();
}
Programação robusta
Caso deseje que as medidas padrão estejam unidades inglesas, implemente os métodos Comprimento e Largura
normalmente e implemente explicitamente os métodos Comprimento e Largura da interface IMetricDimensions:
// Normal implementation:
public float Length() => lengthInches;
public float Width() => widthInches;
// Explicit implementation:
float IMetricDimensions.Length() => lengthInches * 2.54f;
float IMetricDimensions.Width() => widthInches * 2.54f;
Nesse caso, é possível acessar as unidades inglesas da instância de classe e acessar as unidades métricas da
instância da interface:
Consulte também
Guia de Programação em C#
Classes e Structs
Interfaces
Como: implementar membros de interface de forma explícita
Tipos de enumeração (Guia de Programação em
C#)
25/11/2019 • 8 minutes to read • Edit Online
Um tipo de enumeração (também chamado de uma enumeração ou enum) fornece uma maneira eficiente para
definir um conjunto de constantes integrais nomeadas que podem ser atribuídas a um valor. Por exemplo,
suponha que você precisa definir uma variável cujo valor representará um dia da semana. Há apenas sete
valores significativos que essa variável armazenará. Para definir esses valores, você pode usar um tipo de
enumeração, que é declarado usando a palavra-chave enum.
Por padrão, o tipo subjacente de cada elemento na enumeração é int. Você pode especificar outro tipo numérico
integral usando dois-pontos, conforme mostrado no exemplo anterior. Para obter uma lista completa dos tipos
possíveis, consulte enum (Referência de C#).
Você pode verificar os valores numéricos subjacentes com a conversão em tipo subjacente, como mostra o
exemplo a seguir.
// Output:
// Monday is day number #1.
// Dec is month number #11.
NOTE
É possível atribuir qualquer valor inteiro arbitrário a meetingDay . Por exemplo, esta linha de código não produz um erro:
meetingDay = (Day) 42 . No entanto, você não deve fazer isso porque a expectativa implícita é que uma variável enum
conterá apenas um dos valores definidos pela enumeração. Atribuir um valor arbitrário a uma variável de um tipo de
enumeração é introduzir um alto risco de erros.
Você pode atribuir quaisquer valores aos elementos na lista de enumeradores de um tipo de enumeração e
também pode usar valores computados:
enum MachineState
{
PowerOff = 0,
Running = 5,
Sleeping = 10,
Hibernating = Sleeping + 5
}
[Flags]
enum Days
{
None = 0b_0000_0000, // 0
Sunday = 0b_0000_0001, // 1
Monday = 0b_0000_0010, // 2
Tuesday = 0b_0000_0100, // 4
Wednesday = 0b_0000_1000, // 8
Thursday = 0b_0001_0000, // 16
Friday = 0b_0010_0000, // 32
Saturday = 0b_0100_0000 // 64
}
class MyClass
{
Days meetingDays = Days.Tuesday | Days.Thursday;
}
Para definir um sinalizador em uma enumeração, use o operador OR de bit a bit mostrado no exemplo a seguir:
// Initialize with two flags using bitwise OR.
meetingDays = Days.Tuesday | Days.Thursday;
Para determinar se um sinalizador específico é definido, use uma operação AND bit a bit, conforme mostrado no
exemplo a seguir:
Para saber mais sobre o que considerar ao definir tipos de enumeração com o atributo System.FlagsAttribute,
veja System.Enum.
Consulte também
System.Enum
Guia de Programação em C#
enum
Delegados (Guia de Programação em C#)
04/11/2019 • 5 minutes to read • Edit Online
Um delegado é um tipo que representa referências aos métodos com lista de parâmetros e tipo de retorno
específicos. Ao instanciar um delegado, você pode associar sua instância a qualquer método com assinatura e
tipo de retorno compatíveis. Você pode invocar (ou chamar) o método através da instância de delegado.
Delegados são usados para passar métodos como argumentos a outros métodos. Os manipuladores de
eventos nada mais são do que métodos chamados por meio de delegados. Ao criar um método
personalizado, uma classe como um controle do Windows poderá chamá-lo quando um determinado evento
ocorrer. O seguinte exemplo mostra uma declaração de delegado:
Qualquer método de qualquer classe ou struct acessível que corresponda ao tipo delegado pode ser atribuído
ao delegado. O método pode ser estático ou de instância. Isso possibilita alterar via programação chamadas
de método e também conectar novo código a classes existentes.
NOTE
No contexto da sobrecarga de método, a assinatura de um método não inclui o valor retornado. No entanto, no
contexto de delegados, a assinatura inclui o valor retornado. Em outras palavras, um método deve ter o mesmo tipo de
retorno que o delegado.
Essa capacidade de se referir a um método como um parâmetro torna delegados ideais para definir métodos
de retorno de chamada. Por exemplo, uma referência a um método que compara dois objetos poderia ser
passada como um argumento para um algoritmo de classificação. Como o código de comparação está em um
procedimento separado, o algoritmo de classificação pode ser escrito de forma mais geral.
Nesta seção
Usando delegados
Quando usar delegados em vez de interfaces (Guia de Programação em C#)
Delegados com métodos nomeados vs. anônimos
Usando Variação em Delegações
Como combinar delegados (delegados multicast)
Como declarar e usar um delegado e criar uma instância dele
Especificação da Linguagem C#
Para obter mais informações, veja Delegados na Especificação da linguagem C#. A especificação da
linguagem é a fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Delegate
Guia de Programação em C#
Eventos
Usando delegados (Guia de Programação em C#)
04/11/2019 • 10 minutes to read • Edit Online
Um delegado é um tipo que encapsula com segurança um método, semelhante a um ponteiro de função em C e
C++. No entanto, ao contrário dos ponteiros de função de C, delegados são orientados a objeto, fortemente
tipados e seguros. O tipo de um delegado é definido pelo nome do delegado. O exemplo a seguir declara um
delegado chamado Del que pode encapsular um método que usa uma cadeia de caracteres como um argumento
e retorna nulo:
Um objeto delegado é normalmente construído fornecendo-se o nome do método que o delegado encapsulará ou
com uma função anônima. Quando um delegado é instanciado, uma chamada de método feita ao delegado será
passada pelo delegado para esse método. Os parâmetros passados para o delegado pelo chamador são passados
para o método e o valor de retorno, se houver, do método é retornado ao chamador pelo delegado. Isso é
conhecido como invocar o delegado. Um delegado instanciado pode ser invocado como se fosse o método
encapsulado em si. Por exemplo:
Tipos de delegado são derivados da classe Delegate do .NET Framework. Tipos de delegado são lacrados – não
podem ser derivados de – e não é possível derivar classes personalizadas de Delegate. Como o delegado
instanciado é um objeto, ele pode ser passado como um parâmetro ou atribuído a uma propriedade. Isso permite
que um método aceite um delegado como um parâmetro e chame o delegado posteriormente. Isso é conhecido
como um retorno de chamada assíncrono e é um método comum de notificação de um chamador quando um
processo longo for concluído. Quando um delegado é usado dessa maneira, o código que usa o delegado não
precisa de conhecimento algum da implementação do método que está sendo usado. A funcionalidade é
semelhante ao encapsulamento que as interfaces fornecem.
Outro uso comum de chamadas de retorno é definir um método de comparação personalizada e passar esse
delegado para um método de classificação. Ele permite que o código do chamador se torne parte do algoritmo de
classificação. O método de exemplo a seguir usa o tipo Del como um parâmetro:
Em seguida, você pode passar o delegado criado acima para esse método:
MethodWithCallback(1, 2, handler);
Usando o delegado como uma abstração, MethodWithCallback não precisa chamar o console diretamente — ele
não precisa ser criado com um console em mente. O que MethodWithCallback faz é simplesmente preparar uma
cadeia de caracteres e passá-la para outro método. Isso é especialmente poderoso, uma vez que um método
delegado pode usar qualquer número de parâmetros.
Quando um delegado é construído para encapsular um método de instância, o delegado faz referência à instância
e ao método. Um delegado não tem conhecimento do tipo de instância além do método que ele encapsula, de
modo que um delegado pode se referir a qualquer tipo de objeto desde que haja um método nesse objeto que
corresponda à assinatura do delegado. Quando um delegado é construído para encapsular um método estático,
ele só faz referência ao método. Considere as seguintes declarações:
Além do DelegateMethod estático mostrado anteriormente, agora temos três métodos que podem ser
encapsulados por uma instância Del .
Um delegado pode chamar mais de um método quando invocado. Isso é chamado de multicast. Para adicionar um
método extra à lista de métodos do delegado — a lista de invocação — basta adicionar dois delegados usando os
operadores de adição ou de atribuição de adição ('+' ou '+ ='). Por exemplo:
Nesse ponto, allMethodsDelegate contém três métodos em sua lista de invocação — Method1 , Method2 e
DelegateMethod . Os três delegados originais, d1 , d2 e d3 , permanecem inalterados. Quando
allMethodsDelegate é invocado, os três métodos são chamados na ordem. Se o delegado usar parâmetros de
referência, a referência será passada em sequência para cada um dos três métodos por vez, e quaisquer alterações
em um método serão visíveis no próximo método. Quando algum dos métodos gerar uma exceção que não foi
detectada dentro do método, essa exceção será passada ao chamador do delegado e nenhum método subsequente
na lista de invocação será chamado. Se o delegado tiver um valor de retorno e/ou parâmetros de saída, ele
retornará o valor de retorno e os parâmetros do último método invocado. Para remover um método da lista de
invocação, use os operadores de atribuição de subtração ou subtração ( - ou -= ). Por exemplo:
//remove Method1
allMethodsDelegate -= d1;
Como os tipos de delegados são derivados de System.Delegate , os métodos e as propriedades definidos por essa
classe podem ser chamados no delegado. Por exemplo, para localizar o número de métodos na lista de invocação
do delegado, é possível escrever:
Delegados com mais de um método em sua lista de invocação derivam de MulticastDelegate, que é uma subclasse
de System.Delegate . O código acima funciona em ambos os casos, pois as classes oferecem suporte à
GetInvocationList .
Delegados multicast são amplamente usados na manipulação de eventos. Objetos de origem do evento enviam
notificações de eventos aos objetos de destinatário que se registraram para receber esse evento. Para se registrar
para um evento, o destinatário cria um método projetado para lidar com o evento, em seguida, cria um delegado
para esse método e passa o delegado para a origem do evento. A origem chama o delegado quando o evento
ocorre. O delegado chama então o método de manipulação de eventos no destinatário, fornecendo os dados do
evento. O tipo de delegado de um determinado evento é definido pela origem do evento. Para saber mais, consulte
Eventos.
A comparação de delegados de dois tipos diferentes atribuídos no tempo de compilação resultará em um erro de
compilação. Se as instâncias de delegado forem estaticamente do tipo System.Delegate , então a comparação será
permitida, mas retornará false no tempo de execução. Por exemplo:
Consulte também
Guia de Programação em C#
Delegados
Usando Variação em Delegações
Variação em Delegações
Usando Variação para Delegações Genéricas Func e Action
Eventos
Delegados com métodos nomeados versus anônimos
(Guia de Programação em C#)
04/11/2019 • 3 minutes to read • Edit Online
Um delegado pode ser associado a um método nomeado. Ao instanciar um delegado usando um método
nomeado, o método é passado como um parâmetro, por exemplo:
// Declare a delegate.
delegate void Del(int x);
Isso é chamado usando um método nomeado. Os delegados construídos com um método nomeado podem
encapsular um método estático ou um método de instância. Métodos nomeados são a única maneira de instanciar
um delegado nas versões anteriores do C#. No entanto, em uma situação em que a criação de um novo método
for uma sobrecarga indesejada, o C# permite instanciar um delegado e especificar imediatamente um bloco de
código que esse delegado processará quando for chamado. O bloco pode conter uma expressão lambda ou um
método anônimo. Para obter mais informações, consulte Funções Anônimas.
Comentários
O método passado como parâmetro delegado deve ter a mesma assinatura da declaração delegada.
Uma instância de delegado pode encapsular o método estático ou de instância.
Embora o delegado possa usar um parâmetro out, não é recomendável utilizá-lo com delegados de evento
multicast, pois não é possível saber qual delegado será chamado.
Exemplo 1
Este é um exemplo simples de declaração usando um delegado. Observe que tanto o delegado, Del e o método
associado, MultiplyNumbers , têm a mesma assinatura
// Declare a delegate
delegate void Del(int i, double j);
class MathClass
{
static void Main()
{
MathClass m = new MathClass();
Exemplo 2
No exemplo a seguir, um delegado é mapeado para métodos estáticos e de instância e retorna informações
específicas sobre cada um.
// Declare a delegate
delegate void Del();
class SampleClass
{
public void InstanceMethod()
{
Console.WriteLine("A message from the instance method.");
}
class TestSampleClass
{
static void Main()
{
var sc = new SampleClass();
Consulte também
Guia de Programação em C#
Delegados
Como combinar delegados (delegados multicast)
Eventos
Como combinar delegados (delegados multicast)
(Guia de Programação em C#)
04/11/2019 • 2 minutes to read • Edit Online
Este exemplo demonstra como criar delegados multicast. Uma propriedade útil de objetos delegados é que vários
objetos podem ser atribuídos a uma instância delegada usando o operador + . O delegado multicast contém uma
lista dos delegados atribuídos. Quando o delegado multicast é chamado, ele invoca os delegados da lista, em
ordem. Apenas os delegados do mesmo tipo podem ser combinados.
O operador - pode ser usado para remover um delegado de componente de um delegado multicast.
Exemplo
using System;
// Define a custom delegate that has a string parameter and returns void.
delegate void CustomDel(string s);
class TestClass
{
// Define two methods that have the same signature as CustomDel.
static void Hello(string s)
{
Console.WriteLine($" Hello, {s}!");
}
No C# 1.0 e versões posteriores, é possível declarar delegados conforme mostrado no exemplo a seguir.
// Declare a delegate.
delegate void Del(string str);
O C# 2.0 oferece uma maneira mais simples de gravar a declaração anterior, conforme mostrado no exemplo a
seguir.
No C# 2.0 e versões posteriores, também é possível usar um método anônimo para declarar e inicializar um
delegado, conforme mostrado no exemplo a seguir.
No C# 3.0 e versões posteriores, os delegados também podem ser declarados e instanciados usando uma
expressão lambda, conforme mostrado no exemplo a seguir.
Programação robusta
Declarando um delegado.
A instrução a seguir declara um novo tipo de delegado.
Cada tipo de delegado descreve o número e os tipos dos argumentos e o tipo do valor retornado dos
métodos que pode encapsular. Sempre que um novo conjunto de tipos de argumento ou tipo de valor
retornado for necessário, um novo tipo de delegado deverá ser declarado.
Instanciando um delegado.
Após a declaração do tipo de delegado, um objeto delegado deve ser criado e associado a um método
específico. No exemplo anterior, faça isso passando o método PrintTitle para o método
ProcessPaperbackBooks , como no exemplo a seguir:
bookDB.ProcessPaperbackBooks(PrintTitle);
Isso cria um novo objeto delegado associado ao método estático Test.PrintTitle . Da mesma forma, o
método não estático AddBookToTotal no objeto totaller é passado como no exemplo a seguir:
bookDB.ProcessPaperbackBooks(totaller.AddBookToTotal);
processBook(b);
Um delegado pode ser chamado de forma síncrona, como neste exemplo ou de forma assíncrona, usando
os métodos BeginInvoke e EndInvoke .
Consulte também
Guia de Programação em C#
Eventos
Delegados
Matrizes (Guia de Programação em C#)
24/10/2019 • 3 minutes to read • Edit Online
Você pode armazenar diversas variáveis do mesmo tipo em uma estrutura de dados de matriz. Você pode
declarar uma matriz especificando o tipo de seus elementos. Se você quiser que a matriz armazene elementos
de qualquer tipo, você pode especificar object como seu tipo. No sistema de tipos unificado do C#, todos os
tipos, predefinidos e definidos pelo usuário, tipos de referência e tipos de valor, herdam direta ou indiretamente
de Object.
type[] arrayName;
Exemplo
O exemplo a seguir cria matrizes unidimensionais, multidimensionais e denteadas:
class TestArraysClass
{
static void Main()
{
// Declare a single-dimensional array.
int[] array1 = new int[5];
// Alternative syntax.
int[] array3 = { 1, 2, 3, 4, 5, 6 };
// Set the values of the first array in the jagged array structure.
jaggedArray[0] = new int[4] { 1, 2, 3, 4 };
}
}
Seções relacionadas
Matrizes como objetos
Usando foreach com matrizes
Passando matrizes como argumentos
Especificação da linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a fonte
definitiva para a sintaxe e o uso de C#.
Consulte também
Guia de Programação em C#
Coleções
Matrizes como objetos (Guia de Programação em
C#)
24/10/2019 • 2 minutes to read • Edit Online
No C#, as matrizes são objetos e não apenas regiões endereçáveis de memória contígua, como no C e no C++.
Array é o tipo base abstrato de todos os tipos de matriz. Você pode usar as propriedades e outros membros de
classe que Array tem. Um exemplo disso é usar a propriedade Length para obter o comprimento de uma matriz. O
código a seguir atribui o comprimento da matriz numbers , que é 5 , a uma variável denominada lengthOfNumbers :
int[] numbers = { 1, 2, 3, 4, 5 };
int lengthOfNumbers = numbers.Length;
A classe Array fornece vários outros métodos e propriedades úteis para classificar, pesquisar e copiar matrizes.
Exemplo
Este exemplo usa a propriedade Rank para exibir a quantidade de dimensões de uma matriz.
class TestArraysClass
{
static void Main()
{
// Declare and initialize an array.
int[,] theArray = new int[5, 10];
System.Console.WriteLine("The array has {0} dimensions.", theArray.Rank);
}
}
// Output: The array has 2 dimensions.
Consulte também
Guia de Programação em C#
Matrizes
Matrizes unidimensionais
Matrizes multidimensionais
Matrizes denteadas
Matrizes unidimensionais (Guia de Programação em
C#)
04/11/2019 • 3 minutes to read • Edit Online
É possível declarar uma matriz unidimensional de cinco inteiros, conforme mostrado no exemplo a seguir:
Essa matriz contém os elementos de array[0] a array[4] . O operador new é usado para criar a matriz e
inicializar os elementos da matriz para seus valores padrão. Neste exemplo, todos os elementos da matriz são
inicializados em zero.
Uma matriz que armazena os elementos da cadeia de caracteres pode ser declarada da mesma maneira. Por
exemplo:
Inicialização de Matriz
É possível inicializar uma matriz na declaração. Nesse caso, o especificador de comprimento não é necessário
porque ele já é fornecido pelo número de elementos na lista de inicialização. Por exemplo:
Uma matriz de cadeia de caracteres pode ser inicializada da mesma maneira. A seguir, há uma declaração de uma
matriz de cadeia de caracteres em que cada elemento da matriz é inicializado por um nome de um dia:
string[] weekDays = new string[] { "Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat" };
Quando você inicializa uma matriz na declaração, é possível usar os seguintes atalhos:
int[] array2 = { 1, 3, 5, 7, 9 };
string[] weekDays2 = { "Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat" };
É possível declarar uma variável de matriz sem inicialização, mas é necessário usar o operador new quando você
atribui uma matriz a essa variável. Por exemplo:
int[] array3;
array3 = new int[] { 1, 3, 5, 7, 9 }; // OK
//array3 = {1, 3, 5, 7, 9}; // Error
O C# 3.0 introduz matrizes de tipo implícito. Para obter mais informações, consulte Matrizes de tipo implícito.
O resultado dessa instrução depende se SomeType é um tipo de valor ou um tipo de referência. Se for um tipo de
valor, a instrução criará uma matriz de 10 elementos, cada um deles tem o tipo SomeType . Se SomeType for um
tipo de referência, a instrução criará uma matriz de 10 elementos, cada um deles é inicializado com uma
referência nula.
Para obter mais informações sobre tipos de valor e de referência, consulte Tipos.
Consulte também
Array
Guia de Programação em C#
Matrizes
Matrizes multidimensionais
Matrizes denteadas
Matrizes multidimensionais (Guia de Programação
em C#)
23/10/2019 • 3 minutes to read • Edit Online
As matrizes podem ter mais de uma dimensão. Por exemplo, a declaração a seguir cria uma matriz bidimensional
de quatro linhas e duas colunas.
Inicialização de Matriz
É possível inicializar a matriz na declaração, conforme mostrado no exemplo a seguir.
// Two-dimensional array.
int[,] array2D = new int[,] { { 1, 2 }, { 3, 4 }, { 5, 6 }, { 7, 8 } };
// The same array with dimensions specified.
int[,] array2Da = new int[4, 2] { { 1, 2 }, { 3, 4 }, { 5, 6 }, { 7, 8 } };
// A similar array with string elements.
string[,] array2Db = new string[3, 2] { { "one", "two" }, { "three", "four" },
{ "five", "six" } };
// Three-dimensional array.
int[, ,] array3D = new int[,,] { { { 1, 2, 3 }, { 4, 5, 6 } },
{ { 7, 8, 9 }, { 10, 11, 12 } } };
// The same array with dimensions specified.
int[, ,] array3Da = new int[2, 2, 3] { { { 1, 2, 3 }, { 4, 5, 6 } },
{ { 7, 8, 9 }, { 10, 11, 12 } } };
// Output:
// 1
// 2
// 3
// 4
// 7
// three
// 8
// 12
// 12 equals 12
int[,] array4 = { { 1, 2 }, { 3, 4 }, { 5, 6 }, { 7, 8 } };
Caso você escolha declarar uma variável de matriz sem inicialização, será necessário usar o operador new ao
atribuir uma matriz a essa variável. O uso de new é mostrado no exemplo a seguir.
int[,] array5;
array5 = new int[,] { { 1, 2 }, { 3, 4 }, { 5, 6 }, { 7, 8 } }; // OK
//array5 = {{1,2}, {3,4}, {5,6}, {7,8}}; // Error
array5[2, 1] = 25;
Da mesma forma, o exemplo a seguir obtém o valor de um elemento de matriz específico e o atribui à variável
elementValue .
O exemplo de código a seguir inicializa os elementos da matriz com valores padrão (exceto em matrizes
denteadas).
Consulte também
Guia de Programação em C#
Matrizes
Matrizes unidimensionais
Matrizes denteadas
Matrizes denteadas (Guia de Programação em C#)
23/10/2019 • 4 minutes to read • Edit Online
Uma matriz denteada é uma matriz cujos elementos são matrizes. Os elementos de uma matriz denteada podem
ter diferentes dimensões e tamanhos. Às vezes, uma matriz denteada é chamada de uma "matriz de matrizes." Os
exemplos a seguir mostram como declarar, inicializar e acessar matrizes denteadas.
A seguir, há uma declaração de uma matriz unidimensional que tem três elementos, cada um do qual é uma
matriz de dimensão única de inteiros:
Antes de usar jaggedArray , seus elementos devem ser inicializados. É possível inicializar os elementos dessa
forma:
Cada um dos elementos é uma matriz unidimensional de inteiros. O primeiro elemento é uma matriz de 5
inteiros, o segundo é uma matriz de 4 inteiros e o terceiro é uma matriz de 2 inteiros.
Também é possível usar os inicializadores para preencher os elementos matriz com valores, caso em que não é
necessário o tamanho da matriz. Por exemplo:
É possível usar a seguinte forma abreviada. Observe que não é possível omitir o operador new da inicialização
de elementos, porque não há nenhuma inicialização padrão para os elementos:
int[][] jaggedArray3 =
{
new int[] { 1, 3, 5, 7, 9 },
new int[] { 0, 2, 4, 6 },
new int[] { 11, 22 }
};
Uma matriz denteada é uma matriz de matrizes e, portanto, seus elementos são tipos de referência e são
inicializados para null .
É possível acessar elementos de matrizes individuais como estes exemplos:
É possível misturar matrizes denteadas e multidimensionais. A seguir, há uma declaração e inicialização de uma
matriz denteada unidimensional que contém três elementos de matriz bidimensional de tamanhos diferentes.
Para obter mais informações sobre matrizes bidimensionais, consulte Matrizes Multidimensionais.
É possível acessar elementos individuais conforme mostrado neste exemplo, que exibe o valor do elemento
[1,0] da primeira matriz (valor 5 ):
O método Length retorna inúmeros conjuntos contidos na matriz denteada. Por exemplo, supondo que você
tenha declarado a matriz anterior, esta linha:
System.Console.WriteLine(jaggedArray4.Length);
retorna um valor de 3.
Exemplo
Este exemplo cria uma matriz cujos elementos são matrizes. Cada um dos elementos da matriz tem um tamanho
diferente.
class ArrayTest
{
static void Main()
{
// Declare the array of two elements.
int[][] arr = new int[2][];
Consulte também
Array
Guia de Programação em C#
Matrizes
Matrizes unidimensionais
Matrizes multidimensionais
Usar foreach com matrizes (Guia de Programação
em C#)
23/10/2019 • 2 minutes to read • Edit Online
Essa instrução foreach fornece uma maneira simples e limpa de iterar através dos elementos de uma matriz.
Em matrizes unidimensionais, a instrução foreach processa elementos em ordem crescente de índice, começando
com o índice 0 e terminando com índice Length - 1 :
Em matrizes multidimensionais, os elementos são percorridos de modo a que os índices da dimensão mais à
direita sejam aumentados primeiro e, em seguida, da próxima dimensão à esquerda, e assim por diante seguindo
para a esquerda:
No entanto, com matrizes multidimensionais, usar um loop aninhado for oferece mais controle sobre a ordem na
qual processar os elementos da matriz.
Consulte também
Array
Guia de Programação em C#
Matrizes
Matrizes unidimensionais
Matrizes multidimensionais
Matrizes denteadas
Passando matrizes como argumentos (Guia de
Programação em C#)
23/10/2019 • 4 minutes to read • Edit Online
As matrizes podem ser passadas como argumentos para parâmetros de método. Como matrizes são tipos de
referência, o método pode alterar o valor dos elementos.
int[] theArray = { 1, 3, 5, 7, 9 };
PrintArray(theArray);
É possível inicializar e passar uma nova matriz em uma etapa, conforme mostrado no exemplo a seguir.
Exemplo
No exemplo a seguir, uma matriz de cadeia de caracteres é inicializada e passada como um argumento para um
método DisplayArray para cadeias de caracteres. O método exibe os elementos da matriz. Em seguida, o método
ChangeArray inverte os elementos da matriz, e o método ChangeArrayElements modifica os três primeiros
elementos da matriz. Depois que cada método retorna, o método DisplayArray mostra que passar uma matriz por
valor não impede alterações nos elementos da matriz.
using System;
class ArrayExample
{
static void DisplayArray(string[] arr) => Console.WriteLine(string.Join(" ", arr));
int[,] theArray = { { 1, 2 }, { 2, 3 }, { 3, 4 } };
Print2DArray(theArray);
O código a seguir mostra uma declaração parcial de um método de impressão que aceita uma matriz
bidimensional como seu argumento.
void Print2DArray(int[,] arr)
{
// Method code.
}
É possível inicializar e passar uma nova matriz em uma única etapa, conforme é mostrado no exemplo a seguir:
Exemplo
No exemplo a seguir, uma matriz bidimensional de inteiros é inicializada e passada para o método Print2DArray .
O método exibe os elementos da matriz.
class ArrayClass2D
{
static void Print2DArray(int[,] arr)
{
// Display the array elements.
for (int i = 0; i < arr.GetLength(0); i++)
{
for (int j = 0; j < arr.GetLength(1); j++)
{
System.Console.WriteLine("Element({0},{1})={2}", i, j, arr[i, j]);
}
}
}
static void Main()
{
// Pass the array as an argument.
Print2DArray(new int[,] { { 1, 2 }, { 3, 4 }, { 5, 6 }, { 7, 8 } });
Consulte também
Guia de Programação em C#
Matrizes
Matrizes unidimensionais
Matrizes multidimensionais
Matrizes denteadas
Matrizes de tipo implícito (Guia de Programação em
C#)
04/11/2019 • 2 minutes to read • Edit Online
É possível criar uma matriz de tipo implícito na qual o tipo da instância da matriz é inferido com base nos
elementos especificados no inicializador de matriz. As regras para qualquer variável de tipo implícito também se
aplicam a matrizes de tipo implícito. Para obter mais informações, consulte Variáveis locais de tipo implícito.
Geralmente, as matrizes de tipo implícito são usadas em expressões de consulta juntamente com objetos e tipos
anônimos e inicializadores de coleção.
Os exemplos a seguir mostram como criar uma matriz de tipo implícito:
class ImplicitlyTypedArraySample
{
static void Main()
{
var a = new[] { 1, 10, 100, 1000 }; // int[]
var b = new[] { "hello", null, "world" }; // string[]
No exemplo anterior, observe que, com as matrizes de tipo implícito, não são usados colchetes do lado esquerdo
da instrução de inicialização. Observe também que as matrizes denteadas são inicializadas usando new [] assim
como matrizes unidimensionais.
Consulte também
Guia de Programação em C#
Variáveis Locais Tipadas Implicitamente
Matrizes
Tipos Anônimos
Inicializadores de objeto e coleção
var
LINQ em C#
Cadeias de caracteres (Guia de Programação em
C#)
25/11/2019 • 22 minutes to read • Edit Online
Uma cadeia de caracteres é um objeto do tipo String cujo valor é texto. Internamente, o texto é armazenado
como uma coleção sequencial somente leitura de objetos Char. Não há um caractere de finalização null ao fim de
uma cadeia em C#. Portanto, uma cadeia de caracteres em C# pode ter qualquer número de caracteres nulos
inseridos ('\0'). A propriedade Char de uma cadeia de caracteres representa o número de objetos Length que
ela contém e não o número de caracteres Unicode. Para acessar os pontos de código Unicode individuais em
uma cadeia de caracteres, use o objeto StringInfo.
// Initialize to null.
string message2 = null;
Observe que você não usa o operador new para criar um objeto de cadeia de caracteres, exceto ao inicializar a
cadeia de caracteres com uma matriz de caracteres.
Inicialize uma cadeia de caracteres com o valor constante Empty para criar um novo objeto String cuja cadeia de
caracteres tem comprimento zero. A representação de cadeia de caracteres literal de uma cadeia de caracteres de
comprimento zero é "". Ao inicializar cadeias de caracteres com o valor Empty em vez de nulo, você poderá
reduzir as chances de uma NullReferenceException ocorrer. Use o método estático IsNullOrEmpty(String) para
verificar o valor de uma cadeia de caracteres antes de tentar acessá-la.
System.Console.WriteLine(s1);
// Output: A string is more than the sum of its chars.
Como uma cadeia de caracteres de "modificação" na verdade é uma nova criação de cadeia de caracteres, você
deve ter cuidado ao criar referências em cadeias de caracteres. Se você criar uma referência a uma cadeia de
caracteres e "modificar" a cadeia de caracteres original, a referência continuará apontar para o objeto original em
vez do novo objeto que foi criado quando a cadeia de caracteres foi modificada. O código a seguir ilustra esse
comportamento:
System.Console.WriteLine(s2);
//Output: Hello
Para obter mais informações sobre como criar novas cadeias de caracteres com base em modificações, como
operações de pesquisa e substituição na cadeia original, consulte como modificar o conteúdo da cadeia de
caracteres.
Use cadeias de caracteres textuais para conveniência e para facilitar a leitura quando o texto da cadeia de
caracteres contiver caracteres de barra invertida, por exemplo, em caminhos de arquivo. Como as cadeias de
caracteres textuais preservam os caracteres de nova linha como parte do texto da cadeia de caracteres, podem
ser usadas para inicializar cadeias de caracteres de várias linhas. Use aspas duplas para inserir uma marca de
aspas simples dentro de uma cadeia de caracteres textual. O exemplo a seguir mostra alguns usos comuns para
cadeias de caracteres textuais:
string filePath = @"C:\Users\scoleridge\Documents\";
//Output: C:\Users\scoleridge\Documents\
\0 Nulo 0x0000
\a Alerta 0x0007
\b Backspace 0x0008
NOTE
Em tempo de compilação, cadeias de caracteres textuais são convertidas em cadeias de caracteres comuns com as mesmas
sequências de escape. Portanto, se exibir uma cadeia de caracteres textual na janela de observação do depurador, você verá
os caracteres de escape que foram adicionados pelo compilador, não a versão textual do código-fonte. Por exemplo, a
cadeia de caracteres textual @"C:\files.txt" será exibida na janela de inspeção como "C:\\files.txt".
// Output:
// Jupiter Hammon was an African American poet born in 1711.
// He was first published in 1761 at the age of 50.
// He'd be over 300 years old today.
Formatação composta
O String.Format utiliza os espaços reservados entre chaves para criar uma cadeia de caracteres de formato. Este
exemplo resulta em uma saída semelhante para o método de interpolação de cadeia de caracteres usado acima.
var pw = (firstName: "Phillis", lastName: "Wheatley", born: 1753, published: 1773);
Console.WriteLine("{0} {1} was an African American poet born in {2}.", pw.firstName, pw.lastName, pw.born);
Console.WriteLine("She was first published in {0} at the age of {1}.", pw.published, pw.published - pw.born);
Console.WriteLine("She'd be over {0} years old today.", Math.Round((2018d - pw.born) / 100d) * 100d);
// Output:
// Phillis Wheatley was an African American poet born in 1753.
// She was first published in 1773 at the age of 20.
// She'd be over 300 years old today.
Para mais informações sobre formatação de tipos .NET, confira Tipos de formatação em .NET.
Subcadeias de caracteres
Uma subcadeia de caracteres é qualquer sequência de caracteres contida em uma cadeia de caracteres. Use o
método Substring para criar uma nova cadeia de caracteres com base em uma parte da cadeia de caracteres
original. Você pode pesquisar uma ou mais ocorrências de uma subcadeia de caracteres usando o método
IndexOf. Use o método Replace para substituir todas as ocorrências de uma subcadeia de caracteres especificada
por uma nova cadeia de caracteres. Como o método Substring, Replace retorna, na verdade, uma nova cadeia de
caracteres e não a modifica a cadeia de caracteres original. Para obter mais informações, consulte como
Pesquisar cadeias de caracteres e como modificar o conteúdo da cadeia.
System.Console.WriteLine(s3.Replace("C#", "Basic"));
// Output: "Visual Basic Express"
Se os métodos String não fornecerem a funcionalidade necessária para modificar caracteres individuais em uma
cadeia de caracteres, você poderá usar um objeto StringBuilder para modificar os caracteres individuais "in-loco"
e criar uma nova cadeia de caracteres para armazenar os resultados usando os métodos StringBuilder. No
exemplo a seguir, suponha que você deva modificar a cadeia de caracteres original de uma maneira específica e
armazenar os resultados para uso futuro:
string question = "hOW DOES mICROSOFT wORD DEAL WITH THE cAPS lOCK KEY?";
System.Text.StringBuilder sb = new System.Text.StringBuilder(question);
string s = String.Empty;
Por outro lado, uma cadeia de caracteres nula não se refere a uma instância de um objeto System.String e
qualquer tentativa de chamar um método em uma cadeia de caracteres nula provocará uma
NullReferenceException. No entanto, você pode usar cadeias de caracteres nulas em operações de comparação e
concatenação com outras cadeias de caracteres. Os exemplos a seguir ilustram alguns casos em que uma
referência a uma cadeia de caracteres nula faz e não faz com que uma exceção seja lançada:
static void Main()
{
string str = "hello";
string nullStr = null;
string emptyStr = String.Empty;
// The null character can be displayed and counted, like other chars.
string s1 = "\x0" + "abc";
string s2 = "abc" + "\x0";
// Output of the following line: * abc*
Console.WriteLine("*" + s1 + "*");
// Output of the following line: *abc *
Console.WriteLine("*" + s2 + "*");
// Output of the following line: 4
Console.WriteLine(s2.Length);
}
Neste exemplo, um objeto StringBuilder é usado para criar uma cadeia de caracteres com base em um conjunto
de tipos numéricos:
using System;
using System.Text;
namespace CSRefStrings
{
class TestStringBuilder
{
static void Main()
{
var sb = new StringBuilder();
Tópicos relacionados
TÓPICO DESCRIÇÃO
Como modificar o conteúdo de uma cadeia de caracteres Ilustra as técnicas para transformar cadeias de caracteres e
modificar o conteúdo delas.
Como comparar cadeias de caracteres Mostra como executar comparações ordinais e específicas da
cultura de cadeias de caracteres.
Como concatenar várias cadeias de caracteres Demonstra várias maneiras de unir diversas cadeias de
caracteres em uma só.
Como analisar cadeias de caracteres usando String. Split Contém exemplos de código que descrevem como usar o
método String.Split para analisar cadeias de caracteres.
Como pesquisar cadeias de caracteres Explica como usar a pesquisa para texto específico ou
padrões em cadeias de caracteres.
Como determinar se uma cadeia de caracteres representa um Mostra como analisar com segurança uma cadeia de
valor numérico caracteres para ver se ela tem um valor numérico válido.
TÓPICO DESCRIÇÃO
Operações básicas de cadeias de caracteres Fornece links para tópicos que usam os métodos
System.String e System.Text.StringBuilder para executar
operações básicas de cadeia de caracteres.
Como analisar cadeias de caracteres de data e hora no .NET Mostra como converter uma cadeia de caracteres como
"24/01/2008" em um objeto System.DateTime.
LINQ e Cadeias de Caracteres Fornece informações sobre como executar várias operações
de cadeia de caracteres usando consultas LINQ.
Para determinar se uma cadeia de caracteres é uma representação válida de um tipo numérico especificado, use
o método estático TryParse implementado por todos os tipos numéricos primitivos e também por tipos como
DateTime e IPAddress. O exemplo a seguir mostra como determinar se "108" é um int válido.
int i = 0;
string s = "108";
bool result = int.TryParse(s, out i); //i now = 108
Se a cadeia de caracteres contiver caracteres não numéricos ou o valor numérico for muito grande ou muito
pequeno para o tipo especificado, TryParse retornará false e definirá o parâmetro de saída como zero. Caso
contrário, ele retornará true e definirá o parâmetro de saída como o valor numérico da cadeia de caracteres.
NOTE
Uma cadeia de caracteres pode conter apenas caracteres numéricos e ainda não ser válida para o método TryParse do
tipo usado. Por exemplo, "256" não é um valor válido para byte , mas é válido para int . “98,6” não é um valor válido
para int , mas é válido para decimal .
Exemplo
Os exemplos a seguir mostram como usar TryParse com representações de cadeia de caracteres dos valores
long , byte e decimal .
string numString = "1287543"; //"1287543.0" will return false for a long
long number1 = 0;
bool canConvert = long.TryParse(numString, out number1);
if (canConvert == true)
Console.WriteLine("number1 now = {0}", number1);
else
Console.WriteLine("numString is not a valid long");
byte number2 = 0;
numString = "255"; // A value of 256 will return false
canConvert = byte.TryParse(numString, out number2);
if (canConvert == true)
Console.WriteLine("number2 now = {0}", number2);
else
Console.WriteLine("numString is not a valid byte");
decimal number3 = 0;
numString = "27.3"; //"27" is also a valid decimal
canConvert = decimal.TryParse(numString, out number3);
if (canConvert == true)
Console.WriteLine("number3 now = {0}", number3);
else
Console.WriteLine("number3 is not a valid decimal");
Programação robusta
Os tipos numéricos primitivos também implementam o método estático Parse , que lançará uma exceção se a
cadeia de caracteres não for um número válido. Geralmente, TryParse é mais eficiente, pois retornará false
apenas se o número não for válido.
Consulte também
Como: converter uma matriz de bytes em um int
Como: converter uma cadeia de caracteres em um número
Como: converter entre cadeias de caracteres hexadecimais e tipos numéricos
Analisando cadeias de caracteres numéricas
Formatando Tipos
Indexadores (Guia de Programação em C#)
04/11/2019 • 4 minutes to read • Edit Online
Os indexadores permitem que instâncias de uma classe ou struct sejam indexados como matrizes. O valor
indexado pode ser definido ou recuperado sem especificar explicitamente um membro de instância ou tipo.
Os indexadores parecem com propriedades, a diferença é que seus acessadores usam parâmetros.
O exemplo a seguir define uma classe genérica com métodos de acesso get e set simples para atribuir e
recuperar valores. A classe Program cria uma instância dessa classe para armazenar cadeias de caracteres.
using System;
class SampleCollection<T>
{
// Declare an array to store the data elements.
private T[] arr = new T[100];
class Program
{
static void Main()
{
var stringCollection = new SampleCollection<string>();
stringCollection[0] = "Hello, World";
Console.WriteLine(stringCollection[0]);
}
}
// The example displays the following output:
// Hello, World.
NOTE
Para mais exemplos, consulte as seções relacionadas.
class SampleCollection<T>
{
// Declare an array to store the data elements.
private T[] arr = new T[100];
int nextIndex = 0;
class Program
{
static void Main()
{
var stringCollection = new SampleCollection<string>();
stringCollection.Add("Hello, World");
System.Console.WriteLine(stringCollection[0]);
}
}
// The example displays the following output:
// Hello, World.
Observe que => apresenta o corpo da expressão e que a palavra-chave get não é usada.
Começando do C# 7.0, os acessadores get e set podem ser implementados como membros aptos para
expressão. Nesse caso, as palavras-chave get e set devem ser usadas. Por exemplo:
using System;
class SampleCollection<T>
{
// Declare an array to store the data elements.
private T[] arr = new T[100];
class Program
{
static void Main()
{
var stringCollection = new SampleCollection<string>();
stringCollection[0] = "Hello, World.";
Console.WriteLine(stringCollection[0]);
}
}
// The example displays the following output:
// Hello, World.
Visão Geral dos Indexadores
Os indexadores permitem que objetos sejam indexados de maneira semelhante às matrizes.
Um acessador get retorna um valor. Um acessador set atribui um valor.
A palavra-chave this é usada para definir o indexador.
A palavra-chave value é usada para definir o valor que está sendo atribuído pelo indexador set .
Os indexadores não precisam ser indexados por um valor inteiro. Você deve definir o mecanismo de
pesquisa específico.
Os indexadores podem ser sobrecarregados.
Os indexadores podem ter mais de um parâmetro formal, por exemplo, ao acessar uma matriz
bidimensional.
Seções relacionadas
Usando indexadores
Indexadores em interfaces
Comparação entre propriedades e indexadores
Restringindo a acessibilidade ao acessador
Especificação da Linguagem C#
Para obter mais informações, veja Indexadores na Especificação da linguagem C#. A especificação da
linguagem é a fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Guia de Programação em C#
Propriedades
Usando indexadores (Guia de Programação em C#)
23/10/2019 • 8 minutes to read • Edit Online
Os indexadores são uma conveniência sintática que permitem criar uma classe, struct ou interface que os
aplicativos clientes podem acessar como uma matriz. Os indexadores são implementados em tipos cuja principal
finalidade é encapsular uma coleção ou matriz interna. Por exemplo, suponha que você tenha uma classe
TempRecord que representa a temperatura em Fahrenheit, conforme registrada em 10 momentos diferentes
durante um período de 24 horas. A classe contém uma matriz temps do tipo float[] para armazenar os valores
de temperatura. Ao implementar um indexador nessa classe, os clientes podem acessar as temperaturas em uma
instância TempRecord como float temp = tr[4] , e não como float temp = tr.temps[4] . A notação do indexador
não simplifica somente a sintaxe para aplicativos clientes; ela também torna a classe e sua finalidade mais
intuitivas para que os outros desenvolvedores entendam.
Para declarar um indexador em uma classe ou struct, use a palavra-chave this, como mostra o seguinte exemplo:
Comentários
O tipo de um indexador e o tipo dos seus parâmetros devem ser pelo menos tão acessíveis quanto o próprio
indexador. Para obter mais informações sobre níveis de acessibilidade, consulte Modificadores de acesso.
Para obter mais informações sobre como usar indexadores com uma interface, consulte Indexadores de Interface.
A assinatura de um indexador consiste do número e dos tipos de seus parâmetros formais. Ela não inclui o tipo de
indexador nem os nomes dos parâmetros formais. Se você declarar mais de um indexador na mesma classe, eles
terão diferentes assinaturas.
Um valor de indexador não é classificado como uma variável; portanto, não é possível passar um valor de
indexador como um parâmetro ref ou out.
Para fornecer o indexador com um nome que outras linguagens possam usar, use
System.Runtime.CompilerServices.IndexerNameAttribute, como mostra o seguinte exemplo:
[System.Runtime.CompilerServices.IndexerName("TheItem")]
public int this[int index] // Indexer declaration
{
// get and set accessors
}
Este indexador terá o nome TheItem . Não fornecer o atributo de nome tornaria Item o nome padrão.
Exemplo 1
O exemplo a seguir mostra como declarar um campo de matriz privada, temps e um indexador. O indexador
permite acesso direto à instância tempRecord[i] . A alternativa ao uso do indexador é declarar a matriz como um
membro público e acessar seus membros, tempRecord.temps[i] , diretamente.
Observe que, quando o acesso de um indexador é avaliado, por exemplo, em uma instrução Console.Write , o
acessador get é invocado. Portanto, se não existir nenhum acessador get , ocorrerá um erro em tempo de
compilação.
class TempRecord
{
// Array of temperature values
private float[] temps = new float[10] { 56.2F, 56.7F, 56.5F, 56.9F, 58.8F,
61.3F, 65.9F, 62.1F, 59.2F, 57.5F };
set
{
temps[index] = value;
}
}
}
class MainClass
{
static void Main()
{
TempRecord tempRecord = new TempRecord();
// Use the indexer's set accessor
tempRecord[3] = 58.3F;
tempRecord[5] = 60.1F;
}
}
/* Output:
Element #0 = 56.2
Element #1 = 56.7
Element #2 = 56.5
Element #3 = 58.3
Element #4 = 58.8
Element #5 = 60.1
Element #6 = 65.9
Element #7 = 62.1
Element #8 = 59.2
Element #9 = 57.5
*/
Indexando usando outros valores
O C# não limita o tipo de parâmetro do indexador ao inteiro. Por exemplo, talvez seja útil usar uma cadeia de
caracteres com um indexador. Esse indexador pode ser implementado pesquisando a cadeia de caracteres na
coleção e retornando o valor adequado. Como os acessadores podem ser sobrecarregados, as versões do inteiro e
da cadeia de caracteres podem coexistir.
Exemplo 2
O exemplo a seguir declara uma classe que armazena os dias da semana. Um acessador get aceita uma cadeia
de caracteres, o nome de um dia e retorna o inteiro correspondente. Por exemplo, "Sunday" retorna 0, "Monday"
retorna 1 e assim por diante.
// This method finds the day or returns an Exception if the day is not found
private int GetDay(string testDay)
{
class Program
{
static void Main(string[] args)
{
DayCollection week = new DayCollection();
System.Console.WriteLine(week["Fri"]);
// Raises ArgumentOutOfRangeException
System.Console.WriteLine(week["Made-up Day"]);
Programação robusta
Há duas maneiras principais nas quais a segurança e a confiabilidade de indexadores podem ser melhoradas:
Certifique-se de incorporar algum tipo de estratégia de tratamento de erros para manipular a chance de
passagem de código cliente em um valor de índice inválido. Anteriormente, no primeiro exemplo neste
tópico, a classe TempRecord oferece uma propriedade Length que permite que o código cliente verifique a
saída antes de passá-la para o indexador. Também é possível colocador o código de tratamento de erro
dentro do próprio indexador. Certifique-se documentar para os usuários as exceções que você gera dentro
de um acessador do indexador.
Defina a acessibilidade dos acessadores get e set para que ela seja mais restritiva possível. Isso é
importante para o acessador set em particular. Para obter mais informações, consulte Restringindo a
acessibilidade aos acessadores.
Consulte também
Guia de Programação em C#
Indexadores
Propriedades
Indexadores em interfaces (Guia de Programação em
C#)
23/10/2019 • 3 minutes to read • Edit Online
Os indexadores podem ser declarados em uma interface. Acessadores de indexadores de interface diferem dos
acessadores de indexadores de classe das seguintes maneiras:
Os acessadores de interface não usam modificadores.
Um acessador de interface não tem um corpo.
Portanto, a finalidade do acessador é indicar se o indexador é leitura/gravação, somente leitura ou somente
gravação.
Este é um exemplo de um acessador de indexador de interface:
// Indexer declaration:
string this[int index]
{
get;
set;
}
}
A assinatura de um indexador deve ser diferente das assinaturas de todos os outros indexadores declarados na
mesma interface.
Exemplo
O exemplo a seguir mostra como implementar indexadores de interface.
// Indexer on an interface:
public interface ISomeInterface
{
// Indexer declaration:
int this[int index]
{
get;
set;
}
}
class MainClass
{
static void Main()
{
IndexerClass test = new IndexerClass();
System.Random rand = new System.Random();
// Call the indexer to initialize its elements.
for (int i = 0; i < 10; i++)
{
test[i] = rand.Next();
}
for (int i = 0; i < 10; i++)
{
System.Console.WriteLine("Element #{0} = {1}", i, test[i]);
}
No exemplo anterior, é possível usar a implementação de membro de interface explícita usando o nome
totalmente qualificado do membro de interface. Por exemplo:
string ISomeInterface.this[int index]
{
}
No entanto, o nome totalmente qualificado só será necessário para evitar ambiguidade quando a classe estiver
implementando mais de uma interface com a mesma assinatura do indexador. Por exemplo, se uma classe
Employee estiver implementando dois interfaces, ICitizen e IEmployee , e as duas interfaces tiverem a mesma
assinatura de indexador, a implementação de membro de interface explícita é necessária. Ou seja, a seguinte
declaração de indexador:
Consulte também
Guia de Programação em C#
Indexadores
Propriedades
Interfaces
Comparação entre propriedades e indexadores (Guia
de Programação em C#)
23/10/2019 • 2 minutes to read • Edit Online
Os indexadores são como propriedades. Com exceção das diferenças mostradas na tabela a seguir, todas as regras
definidas para acessadores de propriedade também se aplicam a acessadores de indexador.
PROPRIEDADE INDEXADOR
Permite que os métodos sejam chamados como se fossem Permite que elementos de uma coleção interna de um objeto
membros de dados públicos. sejam acessados usando uma notação de matriz no próprio
objeto.
Um acessador get de uma propriedade não tem parâmetros. Um acessador get de um indexador tem a mesma lista de
parâmetro formal que o indexador.
Um acessador set de uma propriedade contém o parâmetro Um acessador set de um indexador tem a mesma lista de
implícito value . parâmetro formal que o indexador, bem como o mesmo
parâmetro de valor.
Dá suporte a sintaxe reduzida com Propriedades Dá suporte a membros aptos para expressão a fim de obter
Autoimplementadas. somente indexadores.
Consulte também
Guia de Programação em C#
Indexadores
Propriedades
Eventos (Guia de Programação em C#)
04/11/2019 • 3 minutes to read • Edit Online
Eventos permitem que uma classe ou objeto notifique outras classes ou objetos quando algo interessante
ocorre. A classe que envia (ou aciona) o evento é chamada de editor e as classes que recebem (ou
manipulam) os eventos são chamadas assinantes.
Em um aplicativo Windows Forms em C# ou Web típico, você assina eventos acionados pelos controles,
como botões e caixas de listagem. Você pode usar o IDE (ambiente de desenvolvimento integrado) do Visual
C# para procurar os eventos que um controle publica e selecionar aqueles que você deseja manipular. O IDE
oferece uma maneira fácil de adicionar automaticamente um método de manipulador de eventos vazio e o
código para assinar o evento. Para obter mais informações, consulte Como Realizar e Cancelar a Assinatura
de Eventos.
Seções relacionadas
Para obter mais informações, consulte:
Como realizar e cancelar a assinatura de eventos
Como publicar eventos em conformidade com as diretrizes do .NET Framework
Como acionar eventos de classe base em classes derivadas
Como implementar eventos de interface
Como implementar acessadores de eventos personalizados
Especificação da Linguagem C#
Para obter mais informações, veja Eventos na Especificação da linguagem C#. A especificação da linguagem
é a fonte definitiva para a sintaxe e o uso de C#.
Capítulos do Livro em Destaque
Expressões lambda, eventos e delegados em C# 3.0 Cookbook, Third Edition: More than 250 solutions for
C# 3.0 programmers
Delegados e eventos em Learning C# 3.0: Master the fundamentals of C# 3.0
Consulte também
EventHandler
Guia de Programação em C#
Delegados
Criando manipuladores de eventos no Windows Forms
Como: realizar e cancelar a assinatura de eventos
(Guia de Programação em C#)
23/10/2019 • 6 minutes to read • Edit Online
Você assina um evento publicado por outra classe quando quer escrever um código personalizado que é
chamado quando esse evento é gerado. Por exemplo, você pode assinar o evento click de um botão para fazer
com que seu aplicativo faça algo útil quando o usuário clicar no botão.
Para assinar eventos usando o IDE do Visual Studio
1. Se você não vir a janela Propriedades, no modo de exibição de Design, clique com o botão direito do
mouse no formulário ou controle para o qual deseja criar um manipulador de eventos e selecione
Propriedades.
2. Na parte superior da janela Propriedades, clique no ícone Eventos.
3. Clique duas vezes no evento que deseja criar, por exemplo, o evento Load .
O Visual C# cria um método de manipulador de eventos vazio e adiciona-o ao código. Como alternativa,
você pode adicionar o código manualmente no modo de exibição Código. Por exemplo, as linhas de
código a seguir declaram um método de manipulador de eventos que será chamado quando a classe
Form gerar o evento Load .
A linha de código que é necessária para assinar o evento também é gerada automaticamente no método
InitializeComponent no arquivo Form1.Designer.cs em seu projeto. Ele é semelhante a isto:
publisher.RaiseCustomEvent += HandleCustomEvent;
Observe que a sintaxe anterior é nova no C# 2.0. Ela é exatamente equivalente à sintaxe C# 1.0, em que o
delegado de encapsulamento deve ser criado explicitamente usando a palavra-chave new :
Você também pode usar uma expressão lambda para especificar um manipulador de eventos:
public Form1()
{
InitializeComponent();
this.Click += (s,e) =>
{
MessageBox.Show(((MouseEventArgs)e).Location.ToString());
};
}
É importante observar que você não pode, com facilidade, cancelar a assinatura de um evento se tiver
usado uma função anônima para assiná-lo. Para cancelar a assinatura nesse cenário, é necessário voltar ao
código em que você assinou o evento, armazenar o método anônimo em uma variável do delegado e,
então, adicionar o delegado ao evento. De modo geral, é recomendável que você não use funções
anônimas para assinar eventos se precisar cancelar a assinatura do evento posteriormente no código. Para
obter mais informações sobre funções anônimas, consulte Funções anônimas.
Cancelando a assinatura
Para impedir que o manipulador de eventos seja invocado quando o evento for gerado, cancele a assinatura do
evento. Para evitar perda de recursos, cancele a assinatura de eventos antes de descartar um objeto de assinante.
Até que você cancele a assinatura de um evento, o delegado multicast subjacente ao evento no objeto de
publicação terá uma referência ao delegado que encapsula o manipulador de eventos do assinante. Desde que o
objeto de publicação contenha essa referência, a coleta de lixo não excluirá seu objeto de assinante.
Para cancelar a assinatura de um evento
Use o operador de atribuição de subtração ( -= ) para cancelar a assinatura de um evento:
publisher.RaiseCustomEvent -= HandleCustomEvent;
Quando todos os assinantes tiverem cancelado a assinatura de um evento, a instância do evento na classe
do publicador será definida como null .
Consulte também
Eventos
event
Como: publicar eventos em conformidade com as diretrizes do .NET Framework
Operadores - e -=
Operadores + e +=
Como: publicar eventos em conformidade com as
diretrizes do .NET Framework (Guia de Programação
em C#)
23/10/2019 • 5 minutes to read • Edit Online
O procedimento a seguir demonstra como adicionar eventos que seguem o padrão do .NET Framework para
classes e structs. Todos os eventos na biblioteca de classes do .NET Framework se baseiam no delegado
EventHandler, que é definido da seguinte maneira:
NOTE
O .NET Framework 2.0 apresenta uma versão genérica desse delegado, o EventHandler<TEventArgs>. Os exemplos a seguir
mostram como usar as duas versões.
Embora os eventos nas classes que você define possam ser baseados em qualquer tipo delegado válido, até
mesmo nos delegados que retornam um valor, é geralmente recomendável que você baseie seus eventos no
padrão do .NET Framework usando o EventHandler, conforme mostrado no exemplo a seguir.
Para publicar eventos com base no padrão EventHandler
1. (Pule esta etapa e vá para a etapa 3a se você não precisa enviar dados personalizados com o evento).
Declare a classe dos seus dados personalizados em um escopo que seja visível para as classes publicadora
e assinante. Em seguida, adicione os membros necessários para manter seus dados de evento
personalizados. Neste exemplo, uma cadeia de caracteres simples é retornada.
2. (Pule esta etapa se você estiver usando a versão genérica de EventHandler<TEventArgs>.) Declare um
delegado na sua classe de publicação. Dê a ela um nome que termina com EventHandler. O segundo
parâmetro especifica o tipo personalizado de EventArgs.
3. Declare o evento em sua classe de publicação, usando uma das etapas a seguir.
a. Se você não tiver uma classe EventArgs personalizada, o tipo de evento será o delegado
EventHandler não genérico. Você não precisa declarar o delegado porque ele já está declarado no
namespace System que está incluído quando você cria seu projeto do C#. Adicione o seguinte
código à sua classe publicadora.
b. Se você estiver usando a versão não genérica de EventHandler e você tem uma classe personalizada
derivada de EventArgs, declare o evento dentro de sua classe de publicação e use o delegado da
etapa 2 como o tipo.
c. Se você estiver usando a versão genérica, não é necessário um delegado personalizado. Em vez
disso, na sua classe de publicação, especifique o tipo de evento como
EventHandler<CustomEventArgs> , substituindo o nome da sua própria classe entre os colchetes
angulares.
Exemplo
O exemplo a seguir demonstra as etapas anteriores, usando uma classe EventArgs personalizada e o
EventHandler<TEventArgs> como o tipo de evento.
namespace DotNetEvents
{
using System;
using System.Collections.Generic;
class Program
{
static void Main(string[] args)
{
Publisher pub = new Publisher();
Subscriber sub1 = new Subscriber("sub1", pub);
Subscriber sub2 = new Subscriber("sub2", pub);
}
}
}
Consulte também
Delegate
Guia de Programação em C#
Eventos
Delegados
Como: acionar eventos de classe base em classes
derivadas (Guia de Programação em C#)
23/10/2019 • 4 minutes to read • Edit Online
O exemplo simples a seguir mostra o modo padrão para declarar os eventos em uma classe base para que eles
também possam ser gerados das classes derivadas. Esse padrão é amplamente usado em classes do Windows
Forms em uma biblioteca de classes do .NET Framework.
Quando você cria uma classe que pode ser usada como uma classe base para outras classes, deve considerar o
fato de que os eventos são um tipo especial de delegado que pode ser invocado apenas de dentro da classe que os
declarou. As classes derivadas não podem invocar diretamente eventos declarados dentro da classe base. Embora,
às vezes, você possa desejar um evento que possa ser gerado apenas pela classe base, na maioria das vezes você
deve habilitar a classe derivada para invocar os eventos de classe base. Para fazer isso, você pode criar um método
de invocação protegido na classe base que encapsula o evento. Chamando ou substituindo esse método de
invocação, as classes derivadas podem invocar o evento diretamente.
NOTE
Não declare eventos virtuais em uma classe base e substitua-os em uma classe derivada. O compilador C# não lida com eles
corretamente e é imprevisível se um assinante do evento derivado realmente estará assinando o evento de classe base.
Exemplo
namespace BaseClassEvents
{
using System;
using System.Collections.Generic;
public ShapeEventArgs(double d)
{
newArea = d;
}
public double NewArea
{
get { return newArea; }
}
}
public ShapeContainer()
{
_list = new List<Shape>();
}
class Test
{
Consulte também
Guia de Programação em C#
Eventos
Delegados
Modificadores de acesso
Criando manipuladores de eventos no Windows Forms
Como: implementar eventos de interface (Guia de
Programação em C#)
23/10/2019 • 4 minutes to read • Edit Online
Um interface pode declarar uma evento. O exemplo a seguir mostra como implementar eventos de interface em
uma classe. Basicamente, as regras são as mesmas aplicadas à implementação de qualquer método ou
propriedade de interface.
namespace ImplementInterfaceEvents
{
public interface IDrawingObject
{
event EventHandler ShapeChanged;
}
public class MyEventArgs : EventArgs
{
// class members
}
public class Shape : IDrawingObject
{
public event EventHandler ShapeChanged;
void ChangeShape()
{
// Do something here before the event…
OnShapeChanged(new MyEventArgs(/*arguments*/));
Exemplo
O exemplo a seguir mostra como lidar com a situação menos comum, na qual a classe herda de duas ou mais
interfaces e cada interface tem um evento com o mesmo nome. Nessa situação, é necessário fornecer uma
implementação explícita da interface para pelo menos um dos eventos. Ao gravar uma implementação explícita
da interface de um evento, também é necessário gravar os acessadores de evento add e remove . Normalmente,
eles são fornecidos pelo compilador, mas nesse caso o compilador não pode fornecê-los.
Ao fornecer acessadores próprios, é possível especificar se os dois eventos são representados pelo mesmo
evento na classe ou por eventos diferentes. Por exemplo, se os eventos forem gerados em horários diferentes, de
acordo com as especificações da interface, será possível associar cada evento a uma implementação separada na
classe. No exemplo a seguir, os assinantes determinam qual evento OnDraw receberão ao converter a referência
de forma para um IShape ou um IDrawingObject .
namespace WrapTwoInterfaceEvents
{
using System;
Console.WriteLine("Drawing a shape.");
}
/* Output:
Sub1 receives the IDrawingObject event.
Drawing a shape.
Sub2 receives the IShape event.
*/
*/
Consulte também
Guia de Programação em C#
Eventos
Delegados
Implementação de interface explícita
Como: acionar eventos de classe base em classes derivadas
Como: implementar acessadores de eventos
personalizados (Guia de Programação em C#)
23/10/2019 • 2 minutes to read • Edit Online
Um evento é um tipo especial de delegado multicast que só pode ser invocado de dentro da classe que ela está
declarado. O código cliente assina o evento ao fornecer uma referência a um método que deve ser invocado
quando o evento for disparado. Esses métodos são adicionados à lista de invocação do delegado por meio de
acessadores de evento, que se assemelham aos acessadores de propriedade, com a exceção de que os acessadores
de eventos são nomeados add e remove . Na maioria dos casos, não é necessário fornecer acessadores de
eventos personalizados. Quando nenhum acessador de evento personalizado for fornecido no código, o
compilador o adicionará automaticamente. No entanto, em alguns casos será necessário fornecer um
comportamento personalizado. Um caso desse tipo é mostrado no tópico Como implementar eventos de
interface.
Exemplo
O exemplo a seguir mostra como implementar os acessadores de eventos personalizados adicionar e remover.
Embora seja possível substituir qualquer código dentro dos acessadores, é recomendável que você bloqueie o
evento antes de adicionar ou remover um novo método de manipulador de eventos.
Consulte também
Eventos
event
Genéricos (Guia de Programação em C#)
30/10/2019 • 6 minutes to read • Edit Online
Os genéricos apresentam o conceito de parâmetros de tipo para o .NET Framework, o que possibilita
criar classes e métodos que adiam a especificação de um ou mais tipos até que a classe ou o método seja
declarado e instanciado pelo código do cliente. Por exemplo, usando um parâmetro de tipo genérico T ,
você pode escrever uma única classe que outro código de cliente pode usar sem incorrer no custo ou
risco de conversões de tempo de execução ou operações de conversão boxing, conforme mostrado aqui:
Classes e métodos genéricos combinam reusabilidade, segurança de tipo e eficiência de forma que suas
contrapartes não genéricas não possam. Os genéricos são usados com mais frequência com coleções e
com os métodos que operam nelas. O namespace System.Collections.Generic contém várias classes de
coleção baseadas em genéricos. As coleções não genéricas, como ArrayList não são recomendadas e são
mantidas para fins de compatibilidade. Para saber mais, confira Genéricos no .NET.
É claro que você também pode criar tipos e métodos genéricos personalizados para fornecer suas
próprias soluções e padrões de design generalizados que sejam fortemente tipados e eficientes. O
exemplo de código a seguir mostra uma classe de lista vinculada genérica simples para fins de
demonstração. (Na maioria dos casos, você deve usar a classe List<T> fornecida pela .NET Framework
biblioteca de classes em vez de criar a sua própria.) O parâmetro de tipo T é usado em vários locais em
que um tipo concreto normalmente seria usado para indicar o tipo do item armazenado na lista. Ele é
usado das seguintes maneiras:
Como o tipo de um parâmetro de método no método AddHead .
Como o tipo de retorno da propriedade Data na classe Node aninhada.
Como o tipo de data do membro particular na classe aninhada.
Observe que T está disponível para a classe Node aninhada. Quando GenericList<T> é instanciada com
um tipo concreto, por exemplo como um GenericList<int> , cada ocorrência de T será substituída por
int .
// type parameter T in angle brackets
public class GenericList<T>
{
// The nested class is also generic on T.
private class Node
{
// T used in non-generic constructor.
public Node(T t)
{
next = null;
data = t;
}
// constructor
public GenericList()
{
head = null;
}
O exemplo de código a seguir mostra como o código cliente usa a classe GenericList<T> genérica para
criar uma lista de inteiros. Ao simplesmente alterar o argumento de tipo, o código a seguir poderia
facilmente ser modificado para criar listas de cadeias de caracteres ou qualquer outro tipo personalizado:
class TestGenericList
{
static void Main()
{
// int is the type argument
GenericList<int> list = new GenericList<int>();
Seções relacionadas
Parâmetros de tipo genérico
Restrições a parâmetros de tipo
Classes genéricas
Interfaces genéricas
Métodos genéricos
Delegados genéricos
Diferenças entre modelos C++ e genéricos C#
Genéricos e reflexão
Genéricos em tempo de execução
Especificação da linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#.
Consulte também
System.Collections.Generic
Guia de Programação em C#
Tipos
<typeparam>
<typeparamref>
Genéricos no .NET
Parâmetros de tipo genérico (Guia de Programação
em C#)
31/10/2019 • 3 minutes to read • Edit Online
Na definição de um tipo genérico ou método, parâmetros de tipo são um espaço reservado para um tipo
específico que o um cliente especifica ao criar uma instância do tipo genérico. Uma classe genérica, como
GenericList<T> , listada em Introdução aos Genéricos, não pode ser usada no estado em que se encontra porque
não é realmente um tipo, mas um plano gráfico de um tipo. Para usar GenericList<T> , o código cliente deve
declarar e instanciar um tipo construído, especificando um argumento de tipo entre colchetes. O argumento de
tipo para essa classe específica pode ser qualquer tipo reconhecido pelo compilador. É possível criar qualquer
quantidade de instâncias do tipo construído, cada uma usando um argumento de tipo diferente, da seguinte
maneira:
Em cada uma dessas instâncias de GenericList<T> , todas as ocorrências de T na classe são substituídas em
tempo de execução com o argumento de tipo. Por meio dessa substituição, cria-se três objetos separados
eficientes e fortemente tipados usando uma única definição de classe. Para obter mais informações sobre como
essa substituição é executada pelo CLR, consulte Genéricos em Tempo de Execução.
Considere usar T como o nome do parâmetro de tipo em tipos com parâmetro de tipo de uma letra.
Considere indicar as restrições colocadas em um parâmetro de tipo no nome do parâmetro. Por exemplo,
um parâmetro restrito a ISession pode ser chamado TSession .
A regra de análise de código CA1715 pode ser usada para garantir que os parâmetros de tipo sejam nomeados
adequadamente.
Consulte também
System.Collections.Generic
Guia de Programação em C#
Genéricos
Diferenças entre modelos C++ e genéricos C#
Restrições a parâmetros de tipo (Guia de
Programação em C#)
27/11/2019 • 16 minutes to read • Edit Online
Restrições informam o compilador sobre as funcionalidades que um argumento de tipo deve ter. Sem nenhuma
restrição, o argumento de tipo poderia ser qualquer tipo. O compilador pode assumir somente os membros de
System.Object, que é a classe base definitiva para qualquer tipo .NET. Para obter mais informações, consulte Por
que usar restrições. Se o código de cliente tentar criar uma instância da classe usando um tipo não permitido por
uma restrição, o resultado será um erro em tempo de compilação. Restrições são especificadas usando a
palavra-chave contextual where . A tabela a seguir lista os sete tipos de restrições:
RESTRIÇÃO DESCRIÇÃO
where T : struct O argumento de tipo deve ser um tipo de valor não anulável.
Para obter informações sobre tipos de valor anulável,
consulte tipos de valor anulável. Como todos os tipos de
valor têm um construtor acessível sem parâmetros, a
restrição struct implica a restrição new() e não pode ser
combinada com a restrição new() . Você também não pode
combinar a restrição de struct com a restrição
unmanaged .
where T : <nome de classe base> O argumento de tipo deve ser ou derivar da classe base
especificada.
public T FindFirstOccurrence(string s)
{
Node current = head;
T t = null;
A restrição permite que a classe genérica use a propriedade Employee.Name . A restrição especifica que todos os
itens do tipo T são um objeto Employee ou um objeto que herda de Employee .
Várias restrições podem ser aplicadas ao mesmo parâmetro de tipo e as restrições em si podem ser tipos
genéricos, da seguinte maneira:
class EmployeeList<T> where T : Employee, IEmployee, System.IComparable<T>, new()
{
// ...
}
Ao aplicar a restrição where T : class , evite os operadores == e != no parâmetro de tipo, pois esses
operadores testarão somente a identidade de referência e não a igualdade de valor. Esse comportamento
ocorrerá mesmo se esses operadores forem sobrecarregados em um tipo usado como argumento. O código a
seguir ilustra esse ponto; a saída é false, muito embora a classe String sobrecarregue o operador == .
O compilador só sabe que T é um tipo de referência em tempo de compilação e deve usar os operadores
padrão que são válidos para todos os tipos de referência. Caso seja necessário testar a igualdade de valor, a
maneira recomendada é também aplicar a restrição where T : IEquatable<T> ou where T : IComparable<T> e
implementar a interface em qualquer classe que seja usada para construir a classe genérica.
class Base { }
class Test<T, U>
where U : struct
where T : Base, new()
{ }
No exemplo anterior, T é uma restrição de tipo no contexto do método Add e um parâmetro de tipo não
associado no contexto da classe List .
Parâmetros de tipo também podem ser usados como restrições em definições de classe genérica. O parâmetro
de tipo deve ser declarado entre colchetes angulares junto com quaisquer outros parâmetros de tipo:
A utilidade dos parâmetros de tipo como restrições com classes genéricas é limitada, pois o compilador não
pode presumir nada sobre o parâmetro de tipo, exceto que ele deriva de System.Object . Use parâmetros de tipo
como restrições em classes genéricas em cenários nos quais deseja impor uma relação de herança entre dois
parâmetros de tipo.
O método anterior deve ser compilado em um contexto unsafe porque ele usa o operador sizeof em um tipo
não conhecido como um tipo interno. Sem a restrição unmanaged , o operador sizeof não está disponível.
A restrição unmanaged implica a restrição struct e não pode ser combinada com ela. Como a restrição struct
implica a restrição new() , a restrição unmanaged também não pode ser combinada com a restrição new() .
Restrições de delegado
Também começando com o C# 7.3, você pode usar System.Delegate ou System.MulticastDelegate como uma
restrição de classe base. O CLR sempre permitia essa restrição, mas a linguagem C# não a permite. A restrição
System.Delegate permite que você escreva código que funcione com delegados de uma maneira fortemente
tipada. O código a seguir define um método de extensão que combina dois delegados, desde que eles sejam do
mesmo tipo:
Você pode usar o método acima para combinar delegados que são do mesmo tipo:
Se você remover a marca de comentário na última linha, ela não será compilada. Tanto first quanto test são
tipos delegados, mas são tipos delegados diferentes.
Restrições de enum
Começando com o C# 7.3, você também pode especificar o tipo System.Enum como uma restrição de classe
base. O CLR sempre permitia essa restrição, mas a linguagem C# não a permite. Genéricos usando System.Enum
fornecem programação fortemente tipada para armazenar em cache os resultados do uso de métodos estáticos
em System.Enum . O exemplo a seguir localiza todos os valores válidos para um tipo enum e, em seguida, cria um
dicionário que mapeia esses valores para sua representação de cadeia de caracteres.
Os métodos usados fazem uso de reflexão, o que tem implicações de desempenho. Você pode chamar esse
método para criar uma coleção que é armazenada em cache e reutilizada, em vez de repetir as chamadas que
exigem reflexão.
Você pode usá-lo conforme mostrado no exemplo a seguir para criar uma enum e compilar um dicionário de
seus nomes e valores:
enum Rainbow
{
Red,
Orange,
Yellow,
Green,
Blue,
Indigo,
Violet
}
Consulte também
System.Collections.Generic
Guia de Programação em C#
Introdução aos genéricos
Classes genéricas
Restrição new
Classes genéricas (Guia de Programação em C#)
23/10/2019 • 7 minutes to read • Edit Online
As classes genéricas encapsulam operações que não são específicas de um determinado tipo de dados. O uso
mais comum das classes genéricas é com coleções, como listas vinculadas, tabelas de hash, pilhas, filas, árvores e
assim por diante. As operações como adicionar e remover itens da coleção são realizadas basicamente da mesma
maneira, independentemente do tipo de dados que estão sendo armazenados.
Na maioria dos cenários que exigem classes de coleção, a abordagem recomendada é usar as que são fornecidas
na biblioteca de classes do .NET. Para obter mais informações sobre o uso dessas classes, consulte Coleções
genéricas no .NET.
Em geral, você cria classes genéricas iniciando com uma classe concreta existente e alterando os tipos para
parâmetros de tipo, um por vez, até alcançar o equilíbrio ideal de generalização e usabilidade. Ao criar suas
próprias classes genéricas, observe as seguintes considerações importantes:
Quais tipos generalizar em parâmetros de tipo.
Como uma regra, quanto mais tipos você puder parametrizar, mais flexível e reutilizável seu código se
tornará. No entanto, generalização em excesso poderá criar um código que seja difícil de ser lido ou
entendido por outros desenvolvedores.
Quais restrições, se houver, aplicar aos parâmetros de tipo (consulte Restrições a parâmetros de tipo).
Uma boa regra é aplicar o máximo de restrições, de maneira que ainda seja possível manipular os tipos
que você precisa manipular. Por exemplo, se você souber que a classe genérica é destinada a ser usada
apenas com tipos de referência, aplique a restrição da classe. Isso impedirá o uso não intencional de sua
classe com tipos de valor e permitirá que você use o operador as em T e verificar se há valores nulos.
Se deve-se levar em consideração o comportamento genérico em subclasses e classes base.
Como as classes genéricas podem servir como classes base, as mesmas considerações de design aplicam-
se nesse caso, como com as classes não genéricas. Consulte as regras sobre heranças de classes base
genéricas mais adiante neste tópico.
Se implementar uma ou mais interfaces genéricas.
Por exemplo, se você estiver projetando uma classe que será usada para criar itens em uma coleção com
base em classes genéricas, poderá ser necessário implementar uma interface como a IComparable<T>, em
que T é o tipo de sua classe.
Para obter um exemplo de uma classe genérica simples, consulte Introdução aos genéricos.
As regras para parâmetros de tipo e restrições têm várias implicações para o comportamento de classes
genéricas, especialmente em relação à acessibilidade de membro e herança. Antes de prosseguir, você deve
compreender alguns termos. Para uma classe genérica Node<T>, , o código cliente pode fazer referência à classe,
especificando um argumento de tipo para criar um tipo construído fechado ( Node<int> ). Como alternativa, ele
pode deixar o parâmetro de tipo não especificado para criar um tipo construído aberto ( Node<T> ), como ao
especificar uma classe base genérica. As classes genéricas podem herdar de classes base construídas concretas,
fechadas ou abertas:
class BaseNode { }
class BaseNodeGeneric<T> { }
// concrete type
class NodeConcrete<T> : BaseNode { }
As classes não genéricas ou em outras palavras, classes concretas, podem herdar de classes base construídas
fechadas, mas não de classes construídas abertas ou de parâmetros de tipo, porque não há maneiras de o código
cliente fornecer o argumento de tipo necessário para instanciar a classe base em tempo de execução.
//No error
class Node1 : BaseNodeGeneric<int> { }
//Generates an error
//class Node2 : BaseNodeGeneric<T> {}
//Generates an error
//class Node3 : T {}
As classes genéricas que herdam de tipos construídos abertos devem fornecer argumentos de tipo para qualquer
parâmetro de tipo de classe base que não é compartilhado pela classe herdeira, conforme demonstrado no código
a seguir:
//No error
class Node4<T> : BaseNodeMultiple<T, int> { }
//No error
class Node5<T, U> : BaseNodeMultiple<T, U> { }
//Generates an error
//class Node6<T> : BaseNodeMultiple<T, U> {}
As classes genéricas que herdam de tipos construídos abertos devem especificar restrições que são um
superconjunto ou sugerem, as restrições no tipo base:
Os tipos genéricos podem usar vários parâmetros de tipo e restrições, da seguinte maneira:
Tipos construídos abertos e construídos fechados podem ser usados como parâmetros de método:
void Swap<T>(List<T> list1, List<T> list2)
{
//code to swap items
}
Se uma classe genérica implementa uma interface, todas as instâncias dessa classe podem ser convertidas nessa
interface.
As classes genéricas são invariáveis. Em outras palavras, se um parâmetro de entrada especifica um
List<BaseClass> , você receberá um erro em tempo de compilação se tentar fornecer um List<DerivedClass> .
Consulte também
System.Collections.Generic
Guia de Programação em C#
Genéricos
Salvar o estado de enumeradores
Um enigma de herança, parte 1
Interfaces genéricas (Guia de Programação em C#)
23/10/2019 • 6 minutes to read • Edit Online
Muitas vezes, é útil definir interfaces para classes de coleção genéricas ou para as classe genéricas que
representam itens na coleção. Para classes genéricas, prefere-se usar interfaces genéricas, como IComparable<T>
em vez de IComparable, a fim de evitar operações de conversão boxing e unboxing em tipos de valor. A biblioteca
de classes .NET Framework define várias interfaces genéricas para uso com as classes de coleção no namespace
System.Collections.Generic.
Quando uma interface é especificada como uma restrição em um parâmetro de tipo, somente os tipos que
implementam a interface podem ser usados. O exemplo de código a seguir mostra uma classe SortedList<T> que
deriva da classe GenericList<T> . Para obter mais informações, consulte Introdução aos Genéricos. SortedList<T>
adiciona a restrição where T : IComparable<T> . Isso habilita o método BubbleSort em SortedList<T> a usar o
método genérico CompareTo em elementos de lista. Neste exemplo, os elementos de lista são uma classe simples,
Person , que implementa IComparable<Person> .
do
{
Node previous = null;
Node current = head;
swapped = false;
if (previous == null)
{
head = tmp;
}
else
{
previous.next = tmp;
}
previous = tmp;
swapped = true;
}
else
{
previous = current;
current = current.next;
}
}
} while (swapped);
}
}
class Program
{
static void Main()
{
//Declare and instantiate a new generic SortedList class.
//Person is the type argument.
SortedList<Person> list = new SortedList<Person>();
int[] ages = new int[] { 45, 19, 28, 23, 18, 9, 108, 72, 30, 35 };
Várias interfaces podem ser especificadas como restrições em um único tipo, da seguinte maneira:
interface IMonth<T> { }
Interfaces genéricas poderão herdar de interfaces não genéricas se a interface genérica for contravariante, o que
significa que ela usa apenas seu parâmetro de tipo como um valor retornado. Na biblioteca de classes do .NET
Framework, IEnumerable<T> herda de IEnumerable porque IEnumerable<T> usa apenas T no valor retornado
de GetEnumerator e no getter de propriedade Current.
Classes concretas podem implementar interfaces construídas fechadas, da seguinte maneira:
interface IBaseInterface<T> { }
Classes genéricas podem implementar interfaces genéricas ou interfaces construídas fechadas, contanto que a
lista de parâmetros de classe forneça todos os argumentos exigidos pela interface, da seguinte maneira:
interface IBaseInterface1<T> { }
interface IBaseInterface2<T, U> { }
As regras que controlam a sobrecarga de método são as mesmas para métodos em classes genéricas, structs
genéricos ou interfaces genéricas. Para obter mais informações, consulte Métodos Genéricos.
Consulte também
Guia de Programação em C#
Introdução aos genéricos
interface
Genéricos
Métodos genéricos (Guia de Programação em C#)
23/10/2019 • 4 minutes to read • Edit Online
O exemplo de código a seguir mostra uma maneira de chamar o método usando int para o argumento de tipo:
Também é possível omitir o argumento de tipo e o compilador o inferirá. Esta chamada para Swap é equivalente à
chamada anterior:
As mesmas regras de inferência de tipos se aplicam a métodos estáticos e métodos de instância. O compilador
pode inferir os parâmetros de tipo com base nos argumentos de método passados; não é possível inferir os
parâmetros de tipo somente de uma restrição ou valor retornado. Portanto, a inferência de tipos não funciona
com métodos que não têm parâmetros. A inferência de tipos ocorre em tempo de compilação, antes de o
compilador tentar resolver assinaturas de método sobrecarregadas. O compilador aplica a lógica da inferência de
tipos a todos os métodos genéricos que compartilham o mesmo nome. Na etapa de resolução de sobrecarga, o
compilador incluirá somente os métodos genéricos em que a inferência de tipos foi bem-sucedida.
Em uma classe genérica, métodos não genéricos podem acessar os parâmetros de tipo de nível de classe, da
seguinte maneira:
class SampleClass<T>
{
void Swap(ref T lhs, ref T rhs) { }
}
Se um método genérico que usa os mesmos parâmetros de tipo da classe que o contém for definido, o
compilador gerará um aviso CS0693, pois, dentro do escopo do método, o argumento fornecido para o T
interno oculta o argumento fornecido para o T externo. Caso seja necessária a flexibilidade de chamar um
método de classe genérica com argumentos de tipo diferentes dos fornecidos quando a instância da classe foi
criada, considere fornecer outro identificador ao parâmetro de tipo do método, conforme mostrado no
GenericList2<T> do exemplo a seguir.
class GenericList<T>
{
// CS0693
void SampleMethod<T>() { }
}
class GenericList2<T>
{
//No warning
void SampleMethod<U>() { }
}
Use restrições para permitir operações mais especializadas em parâmetros de tipo de métodos. Essa versão do
Swap<T> , agora denominada SwapIfGreater<T> , pode ser usada somente com argumentos de tipo que
implementam IComparable<T>.
Métodos genéricos podem ser sobrecarregados vários parâmetros de tipo. Por exemplo, todos os seguintes
métodos podem ser localizados na mesma classe:
void DoWork() { }
void DoWork<T>() { }
void DoWork<T, U>() { }
Especificação da Linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#.
Consulte também
System.Collections.Generic
Guia de Programação em C#
Introdução aos genéricos
Métodos
Genéricos e matrizes (Guia de Programação em C#)
23/10/2019 • 2 minutes to read • Edit Online
No C# 2.0 e versões posteriores, matrizes unidimensionais que têm um limite inferior a zero implementam
IList<T> automaticamente. Isso permite a criação de métodos genéricos que podem usar o mesmo código para
iterar por meio de matrizes e outros tipos de coleção. Essa técnica é útil principalmente para ler dados em coleções.
A interface IList<T> não pode ser usada para adicionar ou remover elementos de uma matriz. Uma exceção será
lançada se você tentar chamar um método IList<T> tal como RemoveAt em uma matriz neste contexto.
O exemplo de código a seguir demonstra como um único método genérico que usa um parâmetro de entrada
IList<T> pode iterar por meio de uma lista e uma matriz, nesse caso, uma matriz de inteiros.
class Program
{
static void Main()
{
int[] arr = { 0, 1, 2, 3, 4 };
List<int> list = new List<int>();
ProcessItems<int>(arr);
ProcessItems<int>(list);
}
Consulte também
System.Collections.Generic
Guia de Programação em C#
Genéricos
Matrizes
Genéricos
Delegados genéricos (Guia de Programação em C#)
04/11/2019 • 2 minutes to read • Edit Online
Um delegado pode definir seus próprios parâmetros de tipo. O código que referencia o delegado genérico pode
especificar o argumento de tipo para criar um tipo construído fechado, assim como quando uma classe genérica é
instanciada ou quando um método genérico é chamado, conforme mostrado no exemplo a seguir:
A versão 2.0 do C# tem um novo recurso chamado conversão de grupo de método, que pode ser aplicada a tipos
concretos e de delegado genérico e habilita a gravação da linha anterior com esta sintaxe simplificada:
Del<int> m2 = Notify;
Os delegados definidos em uma classe genérica podem usar os parâmetros de tipo da classe genérica da mesma
forma que os métodos da classe.
class Stack<T>
{
T[] items;
int index;
O código que referencia o delegado deve especificar o argumento de tipo da classe recipiente, da seguinte
maneira:
Os delegados genéricos são especialmente úteis na definição de eventos com base no padrão de design comum,
pois o argumento do remetente pode ser fortemente tipado e não precisa ser convertido de e para Object.
delegate void StackEventHandler<T, U>(T sender, U eventArgs);
class Stack<T>
{
public class StackEventArgs : System.EventArgs { }
public event StackEventHandler<Stack<T>, StackEventArgs> stackEvent;
class SampleClass
{
public void HandleStackChange<T>(Stack<T> stack, Stack<T>.StackEventArgs args) { }
}
Consulte também
System.Collections.Generic
Guia de Programação em C#
Introdução aos genéricos
Métodos genéricos
Classes genéricas
Interfaces genéricas
Delegados
Genéricos
Diferenças entre modelos C++ e genéricos C# (Guia
de Programação em C#)
31/10/2019 • 3 minutes to read • Edit Online
Os modelos C++ e genéricos C# são recursos de linguagem que fornecem o suporte aos tipos parametrizados.
No entanto, há várias diferenças entre os dois. No nível de sintaxe, os genéricos C# são uma abordagem mais
simples para os tipos parametrizados sem a complexidade de modelos C++. Além disso, o C# não tenta fornecer
toda a funcionalidade que os modelos C++ fornecem. No nível da implementação, a principal diferença é que as
substituições do tipo genérico do C# são realizadas em tempo de execução e as informações do tipo genérico são
preservadas para objetos instanciados. Para obter mais informações, consulte Genéricos em tempo de execução.
A seguir estão as principais diferenças entre modelos C++ e genéricos C#:
Os genéricos C# não oferecem a mesma flexibilidade que os modelos C++. Por exemplo, não é possível
chamar os operadores aritméticos em uma classe genérica C#, embora seja possível chamar operadores
definidos pelo usuário.
O C# não permite parâmetros de modelo sem tipo, como template C<int i> {} .
O C# não dá suporte à especialização explícita ou seja, uma implementação personalizada de um modelo
para um tipo específico.
O C# não dá suporte à especialização parcial: uma implementação personalizada para um subconjunto dos
argumentos de tipo.
O C# não permite que o parâmetro de tipo a ser usado como a classe base para o tipo genérico.
O C# não permite que os parâmetros de tipo tenham tipos padrão.
No C#, um parâmetro de tipo genérico não pode ser genérico, embora os tipos construídos possam ser
usados como genéricos. O C++ permite parâmetros de modelo.
O C++ permite o código que pode não ser válido para todos os parâmetros de tipo no modelo, que é então
verificado para o tipo específico usado como o parâmetro de tipo. O C# requer código em uma classe a ser
gravada de forma que ele funcionará com qualquer tipo que satisfaça as restrições. Por exemplo, em C++ é
possível escrever uma função que usa os operadores aritméticos + e - em objetos do parâmetro de tipo,
que produzirá um erro no momento da instanciação do modelo com um tipo que não dá suporte a esses
operadores. O C# não permite isso. Os únicos constructos da linguagem permitidos são os que podem ser
deduzidos das restrições.
Consulte também
Guia de Programação em C#
Introdução aos genéricos
Modelos
Genéricos em tempo de execução (Guia de
Programação em C#)
31/10/2019 • 6 minutes to read • Edit Online
Um tipo genérico ou método compilado em Microsoft Intermediate Language (MSIL ) conterá metadados que o
identificarão como possuidor de parâmetros de tipo. O uso de MSIL em um tipo genérico será diferente de
acordo com o parâmetro de tipo fornecido, ou seja, se ele é um tipo de valor ou um tipo de referência.
Quando um tipo genérico é construído pela primeira vez com um tipo de valor como parâmetro, o tempo de
execução cria um tipo genérico especializado com o(s) parâmetro(s) fornecido(s) substituído(s) nos locais
apropriados do MSIL. Os tipos genéricos especializados são criados uma vez para cada tipo de valor único usado
como parâmetro.
Por exemplo, caso o código do programa declarar uma pilha construída de inteiros:
Stack<int> stack;
Neste ponto, o tempo de execução gerará uma versão especializada da classe Stack<T> com o inteiro substituído
corretamente, de acordo com seu parâmetro. Agora, sempre que o código do programa utilizar uma pilha de
inteiros, o tempo de execução reutilizará a classe especializada Stack<T> gerada. No exemplo a seguir, são
criadas duas instâncias de uma pilha de inteiros e eles compartilham uma única instância do código Stack<int> :
No entanto, suponha que outra classe Stack<T> com um tipo de valor diferente – como long ou uma estrutura
definida pelo usuário como parâmetro – foi criada em outro ponto do código. Como resultado, o tempo de
execução gerará outra versão do tipo genérico e substituirá um long nos locais apropriados no MSIL.
Conversões não são mais necessárias, pois cada classe genérica especializada contém o tipo de valor
nativamente.
Os genéricos funcionam de outro modo nos tipos de referência. Na primeira vez em que um genérico é
construído com qualquer tipo de referência, o tempo de execução cria um tipo genérico especializado com
referências de objeto substituídas por parâmetros no MSIL. Em seguida, sempre que um tipo construído for
instanciado com um tipo de referência como parâmetro, independentemente do tipo, o tempo de execução
reutilizará a versão especializada do tipo genérico criada anteriormente. Isso é possível porque todas as
referências são do mesmo tamanho.
Por exemplo, suponha que há dois tipos de referência, uma classe Customer e uma classe Order e que uma pilha
de tipos Customer foi criada:
class Customer { }
class Order { }
Stack<Customer> customers;
Neste ponto, o tempo de execução gerará uma versão especializada da classe Stack<T> que armazenará
referências de objeto que serão preenchidas posteriormente, em vez de armazenar dados. Suponha que a
próxima linha de código crie uma pilha de outro tipo de referência, com o nome Order :
Ao contrário dos tipos de valor, outra versão especializada da classe Stack<T> não será criada para o tipo Order .
Em vez disso, uma instância da versão especializada da classe Stack<T> será criada e a variável orders será
definida para referenciá-la. Imagine que uma linha de código foi encontrada para criar uma pilha de um tipo
Customer :
Assim como acontece com o uso anterior da classe Stack<T> criada usando o tipo Order , outra instância da
classe especializada Stack<T> é criada. Os ponteiros contidos nela são definidos para referenciar uma área de
memória do tamanho de um tipo Customer . Como a quantidade de tipos de referência pode variar muito entre
os programas, a implementação de genéricos no C# reduz significativamente a quantidade de código ao diminuir
para um o número de classes especializadas criadas pelo compilador para classes genéricas ou tipos de
referência.
Além disso, quando uma classe genérica do C# for instanciada usando um tipo de valor ou parâmetro de tipo de
referência, a reflexão pode consultá-la em tempo de execução e o seu tipo real e parâmetro de tipo podem ser
determinados.
Consulte também
System.Collections.Generic
Guia de Programação em C#
Introdução aos genéricos
Genéricos
Genéricos e reflexão (Guia de Programação em C#)
23/10/2019 • 5 minutes to read • Edit Online
Como o tempo de execução do CLR (Common Language Runtime) tem acesso às informações de tipo genérico
em tempo de execução, você pode usar a reflexão para obter informações sobre tipos genéricos da mesma forma
como para tipos não genéricos. Para obter mais informações, consulte Genéricos em tempo de execução.
No .NET Framework 2.0, foram adicionados diversos novos membros à classe Type para habilitar as informações
em tempo de execução para tipos genéricos. Consulte a documentação dessas classes para obter mais
informações sobre como usar esses métodos e propriedades. O namespace System.Reflection.Emit também
contém novos membros que dão suporte a genéricos. Confira Como definir um tipo genérico com a emissão de
reflexão.
Para obter uma lista das condições invariáveis para termos usados na reflexão genérica, consulte os comentários
da propriedade IsGenericType.
Além disso, membros da classe MethodInfo habilitam informações em tempo de execução para métodos
genéricos. Consulte os comentários sobre a propriedade IsGenericMethod para obter uma lista das condições
invariáveis para termos usados para refletir sobre os métodos genéricos.
Consulte também
Guia de Programação em C#
Genéricos
Reflexão e tipos genéricos
Genéricos
Genéricos e atributos (Guia de Programação em C#)
23/10/2019 • 2 minutes to read • Edit Online
Atributos podem ser aplicados a tipos genéricos da mesma forma que a tipos não genéricos. Para obter mais
informações sobre a aplicação de atributos, consulte Atributos.
Atributos personalizados são permitidos somente para referenciar tipos genéricos abertos, que são tipos genéricos
para os quais nenhum argumento de tipo é fornecido e tipos genéricos construídos fechados, que fornecem
argumentos para todos os parâmetros de tipo.
Os exemplos a seguir usam esse atributo personalizado:
[CustomAttribute(info = typeof(GenericClass1<>))]
class ClassA { }
Especifica vários parâmetros de tipo usando a quantidade apropriada de vírgulas. Neste exemplo, GenericClass2
tem dois parâmetros de tipo:
[CustomAttribute(info = typeof(GenericClass2<,>))]
class ClassB { }
Um atributo que referencia um parâmetro de tipo genérico causará um erro em tempo de compilação:
Para obter informações sobre um tipo genérico ou um parâmetro de tipo em tempo de execução, é possível usar os
métodos do System.Reflection. Para obter mais informações, consulte Genéricos e Reflexão
Consulte também
Guia de Programação em C#
Genéricos
Atributos
Namespaces (Guia de Programação em C#)
23/10/2019 • 2 minutes to read • Edit Online
Os namespaces são usados intensamente em programações de C# de duas maneiras. Em primeiro lugar, o .NET
Framework usa namespaces para organizar suas muitas classes, da seguinte maneira:
System.Console.WriteLine("Hello World!");
System é um namespace e Console é uma classe nesse namespace. A palavra-chave using pode ser usada para
que o nome completo não seja necessário, como no exemplo a seguir:
using System;
Console.WriteLine("Hello");
Console.WriteLine("World!");
namespace SampleNamespace
{
class SampleClass
{
public void SampleMethod()
{
System.Console.WriteLine(
"SampleMethod inside SampleNamespace");
}
}
}
Especificação da linguagem C#
Para saber mais, confira a seção Namespaces da Especificação da linguagem C#.
Consulte também
Guia de Programação em C#
Usando namespaces
Como: usar o My Namespace
Nomes de identificadores
Diretiva using
:: ??
Usando namespaces (Guia de Programação em C#)
23/10/2019 • 6 minutes to read • Edit Online
Os namespaces são usados intensamente em programas em C# de duas maneiras. Em primeiro lugar, as classes
do .NET Framework usam namespaces para organizar suas muitas classes. Em segundo lugar, declarar seus
próprios namespaces pode ajudar a controlar o escopo dos nomes de classe e de método em projetos de
programação maiores.
Acessar namespaces
A maioria dos aplicativos do C# começam com uma seção de diretivas using . Essa seção lista os namespaces que
o aplicativo usará com frequência e evita que o programador tenha que especificar um nome totalmente
qualificado sempre que um método que está contido é usado.
Por exemplo, ao incluir a linha:
using System;
Console.WriteLine("Hello, World!");
Em vez de:
System.Console.WriteLine("Hello, World!");
aliases de namespace
Você também pode usar a diretiva using para criar um alias para um namespace. Use o qualificador de alias de
namespace :: para acessar os membros do namespace com alias. O exemplo a seguir mostra como criar e usar
um alias de namespace:
using generics = System.Collections.Generic;
namespace AliasExample
{
class TestClass
{
static void Main()
{
generics::Dictionary<string, int> dict = new generics::Dictionary<string, int>()
{
["A"] = 1,
["B"] = 2,
["C"] = 3
};
class Program
{
static void Main(string[] args)
{
// Displays "SampleMethod inside SampleNamespace."
SampleClass outer = new SampleClass();
outer.SampleMethod();
Usando o segmento de código anterior, você pode adicionar um novo membro de classe C3 ao namespace
N1.N2 , da seguinte maneira:
namespace N1.N2
{
class C3 // N1.N2.C3
{
}
}
Em geral, use o qualificador de alias de namespace :: para fazer referência a um alias de namespace ou
global:: para fazer referência ao namespace global e . para qualificar tipos ou membros.
É um erro usar :: com um alias que faz referência a um tipo em vez de um namespace. Por exemplo:
class TestClass
{
static void Main()
{
// Error
//Alias::WriteLine("Hi");
// OK
Alias.WriteLine("Hi");
}
}
Lembre-se que a palavra global não é um alias predefinido, portanto, global.X não tem qualquer significado
especial. Ela só adquire um significado especial quando é usada com :: .
O aviso do compilador CS0440 será gerado se você definir um alias chamado global porque global:: sempre
faz referência ao namespace global e não a um alias. Por exemplo, a linha a seguir gera o aviso:
Usar :: com aliases é uma boa ideia e protege contra a introdução inesperada de tipos adicionais. Considere
este exemplo:
namespace Library
{
public class C : Alias::Exception { }
}
Isso funciona, mas se um tipo nomeado Alias fosse subsequentemente introduzido, Alias. se associaria a esse
tipo. O uso de Alias::Exception garante que Alias seja tratado como um alias de namespace e não seja
confundido com um tipo.
Consulte também
Guia de Programação em C#
Namespaces
Operador .
Operador ::
Alias extern
Como: usar o My Namespace (Guia de Programação
em C#)
23/10/2019 • 2 minutes to read • Edit Online
using Microsoft.VisualBasic.Devices;
Exemplo
Este exemplo chama vários métodos estáticos contidos no namespace MyServices . Para que esse código seja
compilado, deve ser adicionada uma referência ao Microsoft.VisualBasic.DLL ao projeto.
using System;
using Microsoft.VisualBasic.Devices;
class TestMyServices
{
static void Main()
{
// Play a sound with the Audio class:
Audio myAudio = new Audio();
Console.WriteLine("Playing sound...");
myAudio.Play(@"c:\WINDOWS\Media\chimes.wav");
if (myComputer.Network.IsAvailable)
{
Console.WriteLine("Computer is connected to network.");
}
else
{
Console.WriteLine("Computer is not connected to network.");
}
}
}
Nem todas as classes no namespace MyServices podem ser chamadas em um aplicativo C#: por exemplo, a
classe FileSystemProxy não é compatível. Em vez disso, nesse caso específico, podem ser usados os métodos
estáticos que fazem parte da FileSystem, que também estão contidos na VisualBasic.dll. Por exemplo, veja como
usar um desses métodos para duplicar um diretório:
// Duplicate a directory
Microsoft.VisualBasic.FileIO.FileSystem.CopyDirectory(
@"C:\original_directory",
@"C:\copy_of_original_directory");
Consulte também
Guia de Programação em C#
Namespaces
Usando namespaces
Código e ponteiros não seguros (Guia de
Programação em C#)
23/10/2019 • 2 minutes to read • Edit Online
Para manter a segurança de tipos e a segurança, o C# não dá suporte à aritmética de ponteiro por padrão. No
entanto, usando a palavra-chave unsafe, você pode definir um contexto não seguro no qual os ponteiros podem
ser usados. Para obter mais informações sobre ponteiros, confira Tipos de ponteiro.
NOTE
No CLR (Common Language Runtime), o código não seguro é conhecido como código não verificado. O código não seguro
em C# não é necessariamente perigoso. Ele é apenas o código cuja segurança não pode ser verificada pelo CLR. O CLR,
portanto, executará somente o código não seguro se ele estiver em um assembly totalmente confiável. Se você usar código
não seguro, será sua responsabilidade assegurar que seu código não apresente riscos de segurança ou erros de ponteiro.
Seções relacionadas
Para obter mais informações, consulte:
Tipos de ponteiro
Buffers de tamanho fixo
Especificação da linguagem C#
Para obter mais informações, confira o tópico Código não seguro na especificação da linguagem C#.
Consulte também
Guia de Programação em C#
unsafe
Buffers de tamanho fixo (Guia de Programação em
C#)
29/11/2019 • 4 minutes to read • Edit Online
No C#, você pode usar a instrução fixed para criar um buffer com uma matriz de tamanho fixo em uma estrutura
de dados. Os buffers de tamanho fixo são úteis ao escrever métodos que interoperam com fontes de dados de
outras linguagens ou plataformas. A matriz fixa pode usar qualquer atributo ou modificador que for permitido
para membros de struct regulares. A única restrição é que o tipo da matriz deve ser bool , byte , char , short ,
int , long , sbyte , ushort , uint , ulong , float ou double .
Comentários
No código de seguro, um struct C# que contém uma matriz não contém os elementos da matriz. Em vez disso, o
struct contém uma referência aos elementos. Você pode inserir uma matriz de tamanho fixo em um struct quando
ele é usado em um bloco de código não seguro.
O seguinte struct tem 8 bytes de tamanho. A matriz pathName é uma referência:
Um struct pode conter uma matriz inserida em código não seguro. No exemplo a seguir, a matriz fixedBuffer
tem tamanho fixo. É possível uma instrução fixed para estabelecer um ponteiro para o primeiro elemento. Os
elementos da matriz são acessados por este ponteiro. A instrução fixed fixa o campo da instância fixedBuffer
em um local específico na memória.
internal unsafe struct MyBuffer
{
public fixed char fixedBuffer[128];
}
unsafe
{
// Pin the buffer to a fixed location in memory.
fixed (char* charPtr = myC.myBuffer.fixedBuffer)
{
*charPtr = 'A';
}
// Access safely through the index:
char c = myC.myBuffer.fixedBuffer[0];
Console.WriteLine(c);
// modify through the index:
myC.myBuffer.fixedBuffer[0] = 'B';
Console.WriteLine(myC.myBuffer.fixedBuffer[0]);
}
O tamanho da matriz char de 128 elementos é 256 bytes. Buffers de char de tamanho fixo sempre têm dois
bytes por caractere, independentemente da codificação. Isso vale mesmo quando os buffers de char passam por
marshaling para structs ou métodos de API com CharSet = CharSet.Auto ou CharSet = CharSet.Ansi . Para obter
mais informações, consulte CharSet.
O exemplo anterior demonstra o acesso a campos fixed sem fixação, que estão disponíveis a partir do C# 7.3.
Outra matriz de tamanho fixo comum é a matriz bool. Os elementos em uma matriz bool sempre têm um byte
de tamanho. Matrizes bool não são adequadas para criar buffers ou matrizes de bits.
NOTE
Exceto pela memória criada usando stackalloc, o compilador C# e o CLR (Common Language Runtime) não executam
nenhuma verificação de estouro de buffer de segurança. Assim como acontece com qualquer código não seguro, tenha
cuidado.
Consulte também
Guia de Programação em C#
Código não seguro e ponteiros
Instrução fixed
Interoperabilidade
Tipos de ponteiro (Guia de Programação em C#)
04/11/2019 • 6 minutes to read • Edit Online
Em um contexto inseguro, um tipo pode ser de ponteiro, valor ou referência. Uma declaração de tipo de ponteiro
usa uma das seguintes formas:
type* identifier;
void* identifier; //allowed but not recommended
O tipo especificado antes do * em um tipo de ponteiro é chamado de tipo referent. Somente um tipo não
gerenciado pode ser um tipo referent.
Os tipos de ponteiro não são herdados de objeto e não há nenhuma conversão entre tipos de ponteiro e object
. Além disso, as conversões boxing e unboxing não oferecem suporte a ponteiros. No entanto, você pode
converter entre diferentes tipos de ponteiro e tipos de ponteiro e tipos integrais.
Quando você designa vários ponteiros na mesma declaração, o asterisco (*) é escrito junto apenas com o tipo
subjacente; ele não é usado como um prefixo para cada nome de ponteiro. Por exemplo:
Um ponteiro não pode apontar para uma referência ou um struct que contenha referências, pois uma referência
de objeto pode ser coletada como lixo mesmo se um ponteiro estiver apontando para ela. O coletor de lixo não
acompanha se um objeto está sendo apontado por qualquer tipo de ponteiro.
O valor da variável de ponteiro do tipo myType* é o endereço de uma variável do tipo myType . Estes são
exemplos de declarações de tipos de ponteiro:
EXEMPLO DESCRIÇÃO
O operador de indireção de ponteiro * pode ser usado para acessar o conteúdo no local apontado pela variável
de ponteiro. Por exemplo, considere a seguinte declaração:
int* myVariable;
Console.WriteLine("--------");
Console.WriteLine(a[0]);
/*
Output:
10
20
30
--------
10
11
12
--------
12
*/
Você não pode aplicar o operador de indireção para um ponteiro do tipo void* . No entanto, você pode usar
uma conversão para converter um ponteiro nulo em qualquer outro tipo de ponteiro e vice-versa.
Um ponteiro pode ser null . Aplicar o operador de indireção a um ponteiro nulo causa um comportamento
definido por implementação.
Passar ponteiros entre métodos pode causar um comportamento indefinido. Considere usar um método que
retorne um ponteiro para uma variável local por meio de um parâmetro in , out ou ref , ou como o resultado
da função. Se o ponteiro foi definido em um bloco fixo, a variável à qual ele aponta não pode mais ser corrigida.
A tabela a seguir lista os operadores e as instruções que podem operar em ponteiros em um contexto inseguro:
OPERADOR/INSTRUÇÃO USE
[] Indexa um ponteiro.
Para obter mais informações sobre operadores relacionados a ponteiro, veja Operadores relacionados a
ponteiro.
Especificação da linguagem C#
Para saber mais, confira a seção Tipos de ponteiro na Especificação da linguagem C#.
Consulte também
Guia de Programação em C#
Código não seguro e ponteiros
Conversões de ponteiro
Tipos
unsafe
Conversões de ponteiro (Guia de Programação em
C#)
24/11/2019 • 2 minutes to read • Edit Online
A tabela a seguir mostra as conversões de ponteiro implícitas predefinidas. As conversões implícitas podem
ocorrer em diversas situações, incluindo instruções de atribuição e invocações de método.
A conversão explícita de ponteiro é usada para realizar conversões para as quais não há conversão implícita,
usando uma expressão de conversão. A tabela a seguir mostra essas conversões.
sbyte, byte, short, ushort, int, uint, long ou ulong Qualquer tipo de ponteiro
Qualquer tipo de ponteiro sbyte, byte, short, ushort, int, uint, long ou ulong
Exemplo
No exemplo a seguir, um ponteiro para int é convertido em um ponteiro para byte . Observe que o ponteiro
aponta para o menor byte endereçado da variável. Quando você incrementar sucessivamente o resultado, até o
tamanho de int (4 bytes), você poderá exibir os bytes restantes da variável.
unsafe
{
// Convert to byte:
byte* p = (byte*)&number;
Consulte também
Guia de Programação em C#
Tipos de ponteiro
Tipos
unsafe
Instrução fixed
stackalloc
Como: usar ponteiros para copiar uma matriz de
bytes (Guia de Programação em C#)
23/10/2019 • 3 minutes to read • Edit Online
O exemplo a seguir usa ponteiros para copiar bytes de uma matriz para outra.
Este exemplo usa a palavra-chave não seguro, que permite que você use ponteiros no método Copy . A instrução
fixo é usada para declarar ponteiros para as matrizes de origem e de destino. A instrução fixed fixa o local das
matrizes de origem e de destino na memória para que elas não sejam movidas pela coleta de lixo. Os blocos de
memória para as matrizes não serão fixado quando o bloco fixed for concluído. Como o método Copy neste
exemplo usa a palavra-chave unsafe , ele precisa ser compilado com a opção do compilador -unsafe.
Este exemplo acessa os elementos das duas matrizes usando índices em vez de um segundo ponteiro não
gerenciado. A declaração dos ponteiros pSource e pTarget fixa as matrizes. Esse recurso está disponível
começando com o C# 7.3.
Exemplo
static unsafe void Copy(byte[] source, int sourceOffset, byte[] target,
int targetOffset, int count)
{
// If either array is not instantiated, you cannot complete the copy.
if ((source == null) || (target == null))
{
throw new System.ArgumentException();
}
// If the number of bytes from the offset to the end of the array is
// less than the number of bytes you want to copy, you cannot complete
// the copy.
if ((source.Length - sourceOffset < count) ||
(target.Length - targetOffset < count))
{
throw new System.ArgumentException();
}
// The following fixed statement pins the location of the source and
// target objects in memory so that they will not be moved by garbage
// collection.
fixed (byte* pSource = source, pTarget = target)
{
// Copy the specified number of bytes from source to target.
for (int i = 0; i < count; i++)
{
pTarget[targetOffset + i] = pSource[sourceOffset + i];
}
}
}
Consulte também
Guia de Programação em C#
Código não seguro e ponteiros
-unsafe (opções do compilador do C#)
Coleta de lixo
Comentários de documentação XML (Guia de
Programação em C#)
23/10/2019 • 2 minutes to read • Edit Online
No Visual C#, você pode criar documentação para seu código ao incluir elementos XML nos campos de
comentários especiais (indicados por barras triplas) no código-fonte logo antes do bloco de código ao qual os
comentários se referem, por exemplo:
/// <summary>
/// This class performs an important function.
/// </summary>
public class MyClass {}
Quando você compilar com a opção -Doc , o compilador pesquisará todas as marcas XML no código-fonte e
criará um arquivo de documentação XML. Para criar a documentação final com base no arquivo gerado pelo
compilador, crie uma ferramenta personalizada ou use uma ferramenta como o DocFX ou o Sandcastle.
Para consultar elementos XML (por exemplo, sua função processa elementos XML específicos que você deseja
descrever em um comentário da documentação XML ), você pode usar o mecanismo de citação padrão ( < e > ).
Para consultar identificadores genéricos em elementos de referência de código ( cref ), você pode usar os
caracteres de escape (por exemplo, cref="List<T>" ) ou chaves ( cref="List{T}" ). Como um caso especial, o
compilador analisa as chaves como colchetes angulares para tornar o comentário da documentação menos
incômodo para o autor ao fazer referência a identificadores genéricos.
NOTE
Os comentários da documentação XML não são metadados; eles não estão incluídos no assembly compilado e, portanto,
não são acessíveis através de reflexão.
Nesta seção
Marcas recomendadas para comentários de documentação
Processando o Arquivo XML
Delimitadores para marcas de documentação
Como usar os recursos da documentação XML
Seções relacionadas
Para obter mais informações, consulte:
-Doc (comentários de documentação do processo)
Especificação da Linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a fonte
definitiva para a sintaxe e o uso de C#.
Consulte também
Guia de Programação em C#
marcações recomendadas para comentários de
documentação (Guia de Programação em C#)
23/10/2019 • 2 minutes to read • Edit Online
NOTE
Os comentários de documentação não podem ser aplicados a um namespace.
O compilador processará qualquer marca que seja um XML válido. As seguintes marcas fornecem as
funcionalidades geralmente usadas na documentação do usuário.
Marcas
/// <summary>
/// This property always returns a value < 1.
/// </summary>
Consulte também
Guia de Programação em C#
-doc (opções do compilador do C#)
Comentários da documentação XML
Processando o arquivo XML (Guia de Programação
em C#)
25/10/2019 • 9 minutes to read • Edit Online
O compilador gera uma cadeia de identificação para cada constructo no seu código marcado para gerar a
documentação. (Para obter informações sobre como marcar seu código, consulte marcas recomendadas para
comentários de documentação.) A cadeia de caracteres de ID identifica exclusivamente a construção. Programas
que processam o arquivo XML podem usar a cadeia de identificação para identificar o item de metadados/reflexão
do .NET Framework correspondente ao qual a documentação se aplica.
O arquivo XML não é uma representação hierárquica de seu código; é uma lista simples com uma ID gerada para
cada elemento.
O compilador observa as seguintes regras quando gera as cadeias de identificação:
Não há espaços em branco na cadeia de caracteres.
A primeira parte da cadeia de identificação identifica o tipo de membro que está sendo identificado por
meio de um único caractere seguido por dois-pontos. São usados os seguintes tipos de membro:
CARACTERE DESCRIÇÃO
N namespace
F campo
E evento
A segunda parte da cadeia de caracteres é o nome totalmente qualificado do item, iniciando na raiz do
namespace. O nome do item, seus tipos delimitadores e o namespace são separados por pontos. Se o
nome do próprio item tiver pontos, eles serão substituídos pelo sustenido ('#'). Supõe-se que nenhum item
tem um sustenido diretamente em seu nome. Por exemplo, o nome totalmente qualificado do construtor de
cadeia de caracteres seria "System.String.#ctor".
Para propriedades e métodos, se houver argumentos para o método, seguirá a lista de argumentos entre
parênteses. Se não houver nenhum argumento, não haverá parênteses. Os argumentos são separados por
vírgulas. A codificação de cada argumento segue diretamente a maneira como ele é codificado em uma
assinatura do .NET Framework:
Tipos base. Tipos regulares (ELEMENT_TYPE_CLASS ou ELEMENT_TYPE_VALUETYPE ) são
representados como o nome totalmente qualificado do tipo.
Os tipos intrínsecos (por exemplo, ELEMENT_TYPE_I4, ELEMENT_TYPE_OBJECT,
ELEMENT_TYPE_STRING, ELEMENT_TYPE_TYPEDBYREF e ELEMENT_TYPE_VOID ) são
representados como o nome totalmente qualificado do tipo completo correspondente. Por exemplo,
System.Int32 ou System.TypedReference.
ELEMENT_TYPE_PTR é representado como um '*' após o tipo modificado.
ELEMENT_TYPE_BYREF é representado como um "@" após o tipo modificado.
ELEMENT_TYPE_PINNED é representado como um '^' após o tipo modificado. O compilador C#
nunca gera isso.
ELEMENT_TYPE_CMOD_REQ é representado como um '|' e o nome totalmente qualificado da
classe do modificador, após o tipo modificado. O compilador C# nunca gera isso.
ELEMENT_TYPE_CMOD_OPT é representado como um '!' e o nome totalmente qualificado da
classe do modificador, após o tipo modificado.
ELEMENT_TYPE_SZARRAY é representado como "[]" após o tipo de elemento da matriz.
ELEMENT_TYPE_GENERICARRAY é representado como "[?]" após o tipo de elemento da matriz. O
compilador C# nunca gera isso.
ELEMENT_TYPE_ARRAY é representado como [lowerbound: size ,lowerbound: size ] em que o
número de vírgulas é a classificação -1 e os limites e o tamanho inferiores de cada dimensão, se
conhecidos, são representados no formato decimal. Se um limite ou tamanho inferior não for
especificado, ele é simplesmente omitido. Se o limite e o tamanho inferiores de uma determinada
dimensão forem omitidos, o ':' será omitido também. Por exemplo, uma matriz bidimensional com 1
como limites inferiores e tamanhos não especificados é [1:,1:].
ELEMENT_TYPE_FNPTR é representado como "=FUNC: type (assinatura)", em que type é o tipo
de retorno e assinatura são os argumentos do método. Se não houver nenhum argumento, os
parênteses serão omitidos. O compilador C# nunca gera isso.
Os seguintes componentes de assinatura não são representados, porque nunca são usadas para
diferenciar métodos sobrecarregados:
convenção de chamada
tipo de retorno
ELEMENT_TYPE_SENTINEL
Somente para operadores de conversão (op_Implicit e op_Explicit), o valor retornado do método é
codificado como um '~' seguido pelo tipo de retorno, conforme codificado acima.
Para tipos genéricos, o nome do tipo é seguido por um caractere de acento grave e, em seguida, por um
número que indica o número de parâmetros de tipo genérico. Por exemplo:
<member name="T:SampleClass`2"> é a marcação de um tipo definido como public class SampleClass<T, U> .
Para métodos que aceitam tipos genéricos como parâmetros, os parâmetros de tipo genérico são
especificados como números precedidos por caracteres de acento grave (por exemplo `0,`1). Cada número
que representa uma notação de matriz com base em zero para parâmetros genéricos do tipo.
Exemplos
Os exemplos a seguir mostram como as cadeias de identificação para uma classe e seus membros seriam geradas:
namespace N
{
/// <summary>
/// Enter description here for class X.
/// ID string generated is "T:N.X".
/// </summary>
public unsafe class X
{
/// <summary>
/// Enter description here for the first constructor.
/// ID string generated is "M:N.X.#ctor".
/// </summary>
public X() { }
/// <summary>
/// Enter description here for the second constructor.
/// ID string generated is "M:N.X.#ctor(System.Int32)".
/// </summary>
/// <param name="i">Describe parameter.</param>
public X(int i) { }
/// <summary>
/// Enter description here for field q.
/// ID string generated is "F:N.X.q".
/// </summary>
public string q;
/// <summary>
/// Enter description for constant PI.
/// ID string generated is "F:N.X.PI".
/// </summary>
public const double PI = 3.14;
/// <summary>
/// Enter description for method f.
/// ID string generated is "M:N.X.f".
/// </summary>
/// <returns>Describe return value.</returns>
public int f() { return 1; }
/// <summary>
/// Enter description for method bb.
/// ID string generated is "M:N.X.bb(System.String,System.Int32@,System.Void*)".
/// </summary>
/// <param name="s">Describe parameter.</param>
/// <param name="y">Describe parameter.</param>
/// <param name="z">Describe parameter.</param>
/// <returns>Describe return value.</returns>
public int bb(string s, ref int y, void* z) { return 1; }
/// <summary>
/// Enter description for method gg.
/// ID string generated is "M:N.X.gg(System.Int16[],System.Int32[0:,0:])".
/// </summary>
/// </summary>
/// <param name="array1">Describe parameter.</param>
/// <param name="array">Describe parameter.</param>
/// <returns>Describe return value.</returns>
public int gg(short[] array1, int[,] array) { return 0; }
/// <summary>
/// Enter description for operator.
/// ID string generated is "M:N.X.op_Addition(N.X,N.X)".
/// </summary>
/// <param name="x">Describe parameter.</param>
/// <param name="xx">Describe parameter.</param>
/// <returns>Describe return value.</returns>
public static X operator +(X x, X xx) { return x; }
/// <summary>
/// Enter description for property.
/// ID string generated is "P:N.X.prop".
/// </summary>
public int prop { get { return 1; } set { } }
/// <summary>
/// Enter description for event.
/// ID string generated is "E:N.X.d".
/// </summary>
public event D d;
/// <summary>
/// Enter description for property.
/// ID string generated is "P:N.X.Item(System.String)".
/// </summary>
/// <param name="s">Describe parameter.</param>
/// <returns></returns>
public int this[string s] { get { return 1; } }
/// <summary>
/// Enter description for class Nested.
/// ID string generated is "T:N.X.Nested".
/// </summary>
public class Nested { }
/// <summary>
/// Enter description for delegate.
/// ID string generated is "T:N.X.D".
/// </summary>
/// <param name="i">Describe parameter.</param>
public delegate void D(int i);
/// <summary>
/// Enter description for operator.
/// ID string generated is "M:N.X.op_Explicit(N.X)~System.Int32".
/// </summary>
/// <param name="x">Describe parameter.</param>
/// <returns>Describe return value.</returns>
public static explicit operator int(X x) { return 1; }
}
}
Consulte também
Guia de Programação em C#
-doc (opções do compilador do C#)
Comentários da documentação XML
Delimitadores para marcações de documentação
(Guia de Programação em C#)
23/10/2019 • 5 minutes to read • Edit Online
O uso de comentários do documento XML requer delimitadores, que indicam ao compilador em que um
comentário de documentação começa e termina. Você pode usar os seguintes tipos de delimitadores com as
marcas de documentação XML:
///
O delimitador de linha única. Este é o formulário mostrado nos exemplos de documentação e usado pelos
modelos de projeto do Visual C#. Se houver um caractere de espaço em branco depois do delimitador, esse
caractere não estará incluído na saída XML.
NOTE
A IDE do Visual Studio tem um recurso chamado Edição de Comentário Inteligente que insere automaticamente as marcas
<summary> e </summary> e move o cursor dentro dessas marcas depois que você digita o delimitador /// no Editor de
Código. Você pode ativar ou desativar esse recurso na caixa de diálogo Opções.
/** */
Delimitadores multilinha.
Há algumas regras de formatação a serem seguidas ao usar os delimitadores /** */ .
Na linha que contém o delimitador /** , se o restante da linha for um espaço em branco, a linha não será
processada para comentários. Se o primeiro caractere após o delimitador /** for um espaço em branco,
esse caractere de espaço em branco será ignorado e o restante da linha será processado. Caso contrário,
todo o texto da linha após o delimitador /** é processado como parte do comentário.
Na linha que contém o delimitador */ , se houver apenas espaços em branco até o delimitador */ , essa
linha será ignorada. Caso contrário, o texto da linha até o delimitador */ é processado como parte do
comentário, sujeito às regras de correspondência de padrão descritas no marcador seguinte.
Para as linhas após a que começa com o delimitador /** , o compilador procura um padrão comum no
início de cada linha. O padrão pode consistir de espaço em branco opcional e um asterisco ( * ), seguido de
mais espaço em branco opcional. Se o compilador encontrar um padrão comum no início de cada linha que
não começa com o delimitador /** ou com o delimitador */ , ele ignorará esse padrão para cada linha.
Os exemplos a seguir ilustram essas regras.
A única parte do comentário a seguir que será processada é a linha que começa com <summary> . Os três
formatos de marca produzem os mesmos comentários.
/** <summary>text</summary> */
/**
<summary>text</summary>
*/
/**
* <summary>text</summary>
*/
O compilador identifica um padrão comum de " * " no início da segunda e terceira linhas. O padrão não é
incluído na saída.
/**
* <summary>
* text </summary>*/
O compilador não encontra nenhum padrão comum no seguinte comentário porque o segundo caractere
na terceira linha não é um asterisco. Portanto, todo o texto na segunda e terceira linhas é processado como
parte do comentário.
/**
* <summary>
text </summary>
*/
O compilador não encontra nenhum padrão no seguinte comentário por dois motivos. Primeiro, o número
de espaços antes do asterisco não é consistente. Segundo, a quinta linha começa com uma guia, que não
coincide com espaços. Portanto, todo o texto das linhas de dois a cinco é processado como parte do
comentário.
/**
* <summary>
* text
* text2
* </summary>
*/
Consulte também
Guia de Programação em C#
Comentários da documentação XML
-doc (opções do compilador do C#)
Comentários da documentação XML
Como usar as funcionalidades da documentação
XML
23/10/2019 • 6 minutes to read • Edit Online
O exemplo a seguir fornece uma visão geral básica de um tipo que foi documentado.
Exemplo
// If compiling from the command line, compile with: -doc:YourFileName.xml
/// <summary>
/// Class level summary documentation goes here.
/// </summary>
/// <remarks>
/// Longer comments can be associated with a type or member through
/// the remarks tag.
/// </remarks>
public class TestClass : TestInterface
{
/// <summary>
/// Store for the Name property.
/// </summary>
private string _name = null;
/// <summary>
/// The class constructor.
/// </summary>
public TestClass()
{
// TODO: Add Constructor Logic here.
}
/// <summary>
/// Name property.
/// </summary>
/// <value>
/// A value tag is used to describe the property value.
/// </value>
public string Name
{
get
{
if (_name == null)
{
throw new System.Exception("Name is null");
}
return _name;
}
}
/// <summary>
/// Description for SomeMethod.
/// </summary>
/// <param name="s"> Parameter description for s goes here.</param>
/// <seealso cref="System.String">
/// You can use the cref attribute on any tag to reference a type or member
/// and the compiler will check that the reference exists.
/// </seealso>
public void SomeMethod(string s)
{
{
}
/// <summary>
/// Some other method.
/// </summary>
/// <returns>
/// Return values are described through the returns tag.
/// </returns>
/// <seealso cref="SomeMethod(string)">
/// Notice the use of the cref attribute to reference a specific method.
/// </seealso>
public int SomeOtherMethod()
{
return 0;
}
/// <summary>
/// The entry point for the application.
/// </summary>
/// <param name="args"> A list of command line arguments.</param>
static int Main(System.String[] args)
{
// TODO: Add code to start application here.
return 0;
}
}
/// <summary>
/// Documentation that describes the interface goes here.
/// </summary>
/// <remarks>
/// Details about the interface go here.
/// </remarks>
interface TestInterface
{
/// <summary>
/// Documentation that describes the method goes here.
/// </summary>
/// <param name="n">
/// Parameter n requires an integer argument.
/// </param>
/// <returns>
/// The method returns an integer.
/// </returns>
int InterfaceMethod(int n);
}
<?xml version="1.0"?>
<doc>
<assembly>
<name>xmlsample</name>
</assembly>
<members>
<member name="T:TestClass">
<summary>
Class level summary documentation goes here.
</summary>
<remarks>
Longer comments can be associated with a type or member through
the remarks tag.
</remarks>
</member>
<member name="F:TestClass._name">
<summary>
Store for the Name property.
</summary>
</member>
<member name="M:TestClass.#ctor">
<summary>
The class constructor.
</summary>
</member>
<member name="P:TestClass.Name">
<summary>
Name property.
</summary>
<value>
A value tag is used to describe the property value.
</value>
</member>
<member name="M:TestClass.SomeMethod(System.String)">
<summary>
Description for SomeMethod.
</summary>
<param name="s"> Parameter description for s goes here.</param>
<seealso cref="T:System.String">
You can use the cref attribute on any tag to reference a type or member
and the compiler will check that the reference exists.
</seealso>
</member>
<member name="M:TestClass.SomeOtherMethod">
<summary>
Some other method.
</summary>
<returns>
Return values are described through the returns tag.
</returns>
<seealso cref="M:TestClass.SomeMethod(System.String)">
Notice the use of the cref attribute to reference a specific method.
</seealso>
</member>
<member name="M:TestClass.Main(System.String[])">
<summary>
The entry point for the application.
</summary>
<param name="args"> A list of command line arguments.</param>
</member>
<member name="T:TestInterface">
<summary>
Documentation that describes the interface goes here.
</summary>
<remarks>
Details about the interface go here.
</remarks>
</member>
<member name="M:TestInterface.InterfaceMethod(System.Int32)">
<summary>
Documentation that describes the method goes here.
</summary>
<param name="n">
Parameter n requires an integer argument.
</param>
<returns>
The method returns an integer.
</returns>
</member>
</members>
</doc>
Compilando o código
Para compilar o exemplo, digite a seguinte linha de comando:
csc XMLsample.cs /doc:XMLsample.xml
Esse comando cria o arquivo XML XMLsample.xml, que você pode exibir no navegador ou usando o comando
TYPE.
Programação robusta
A documentação XML começa com ///. Quando você cria um novo projeto, os assistentes colocam algumas linhas
iniciais /// para você. O processamento desses comentários tem algumas restrições:
A documentação deve ser em XML bem formado. Se o XML não estiver bem formado, um aviso será
gerado e o arquivo de documentação conterá um comentário que diz que foi encontrado um erro.
Os desenvolvedores são livres para criar seu próprio conjunto de marcas. Há um conjunto de marcas
recomendadas (confira Marcas recomendadas para comentários da documentação). Algumas das marcas
recomendadas têm significado especial:
A marca <param> é usada para descrever parâmetros. Se ela é usada, o compilador verifica se o
parâmetro existe e se todos os parâmetros são descritos na documentação. Se a verificação falhar, o
compilador emitirá um aviso.
O atributo cref pode ser anexado a qualquer marca para fornecer uma referência a um elemento
de código. O compilador verifica se esse elemento de código existe. Se a verificação falhar, o
compilador emitirá um aviso. O compilador respeita qualquer instrução using quando procura por
um tipo descrito no atributo cref .
A marca <summary> é usada pelo IntelliSense no Visual Studio para exibir informações adicionais
sobre um tipo ou membro.
NOTE
O arquivo XML não fornece informações completas sobre o tipo e os membros (por exemplo, ele não contém
nenhuma informação de tipo). Para obter informações completas sobre um tipo ou membro, o arquivo de
documentação deve ser usado com a reflexão no membro ou tipo real.
Consulte também
Guia de Programação em C#
-doc (opções do compilador do C#)
Comentários da documentação XML
Processador de documentação do DocFX
Processador de documentação do Sandcastle
<c> (Guia de Programação em C#)
23/10/2019 • 2 minutes to read • Edit Online
Sintaxe
<c>text</c>
Parâmetros
text
O texto que você deseja indicar como código.
Comentários
A marca <c> oferece uma maneira de indicar que o texto em uma descrição deve ser marcado como código. Use
<code> para indicar várias linhas como código.
Compile com -doc para processar comentários de documentação em um arquivo.
Exemplo
// compile with: -doc:DocFileName.xml
Consulte também
Guia de Programação em C#
Marcas recomendadas para comentários de documentação
<code> (Guia de Programação em C#)
23/10/2019 • 2 minutes to read • Edit Online
Sintaxe
<code>content</code>
Parâmetros
content
O texto a que ser marcado como código.
Comentários
A marca <code> oferece uma maneira de indicar várias linhas como código. Use <c> para indicar que o texto
dentro uma descrição deve ser marcado como código.
Compile com -doc para processar comentários de documentação em um arquivo.
Exemplo
Consulte o tópico <example> para obter um exemplo de como usar a marca <code>.
Consulte também
Guia de Programação em C#
Marcas recomendadas para comentários de documentação
Atributo cref (Guia de Programação em C#)
23/10/2019 • 3 minutes to read • Edit Online
O atributo cref em uma marca de documentação XML significa “referência de código”. Ele especifica que o texto
interno da marca é um elemento de código, como um tipo, método ou propriedade. Ferramentas de
documentação, como o DocFX e o Sandcastle, usam os atributos cref para gerar automaticamente os hiperlinks
para a página em que o tipo ou o membro está documentado.
Exemplo
A exemplo a seguir mostra os atributos cref usados em marcas <see>.
namespace TestNamespace
{
/// <summary>
/// TestClass contains several cref examples.
/// </summary>
public class TestClass
{
/// <summary>
/// This sample shows how to specify the <see cref="TestClass"/> constructor as a cref attribute.
/// </summary>
public TestClass()
{ }
/// <summary>
/// This sample shows how to specify the <see cref="TestClass(int)"/> constructor as a cref attribute.
/// </summary>
public TestClass(int value)
{ }
/// <summary>
/// The GetZero method.
/// </summary>
/// <example>
/// This sample shows how to call the <see cref="GetZero"/> method.
/// <code>
/// class TestClass
/// {
/// static int Main()
/// {
/// return GetZero();
/// }
/// }
/// </code>
/// </example>
public static int GetZero()
{
return 0;
}
/// <summary>
/// The GetGenericValue method.
/// </summary>
/// <remarks>
/// This sample shows how to specify the <see cref="GetGenericValue"/> method as a cref attribute.
/// </remarks>
public static T GetGenericValue<T>(T para)
{
return para;
}
}
/// <summary>
/// GenericClass.
/// </summary>
/// <remarks>
/// This example shows how to specify the <see cref="GenericClass{T}"/> type as a cref attribute.
/// </remarks>
class GenericClass<T>
{
// Fields and members.
}
class Program
{
static int Main()
{
return TestClass.GetZero();
}
}
}
Quando compilado, o programa produz o seguinte arquivo XML. Observe que o atributo cref do método
GetZero , por exemplo, foi transformado em "M:TestNamespace.TestClass.GetZero" pelo compilador. O prefixo "M:"
significa "método" e é uma convenção reconhecida por ferramentas de documentação como o DocFX e o
Sandcastle. Para obter uma lista completa de prefixos, consulte Processando o Arquivo XML.
<?xml version="1.0"?>
<doc>
<assembly>
<name>CRefTest</name>
</assembly>
<members>
<member name="T:TestNamespace.TestClass">
<summary>
TestClass contains cref examples.
</summary>
</member>
<member name="M:TestNamespace.TestClass.#ctor">
<summary>
This sample shows how to specify the <see cref="T:TestNamespace.TestClass"/> constructor as a cref
attribute.
</summary>
</member>
<member name="M:TestNamespace.TestClass.#ctor(System.Int32)">
<summary>
This sample shows how to specify the <see cref="M:TestNamespace.TestClass.#ctor(System.Int32)"/>
constructor as a cref attribute.
</summary>
</member>
<member name="M:TestNamespace.TestClass.GetZero">
<summary>
The GetZero method.
</summary>
<example>
This sample shows how to call the <see cref="M:TestNamespace.TestClass.GetZero"/> method.
<code>
class TestClass
{
static int Main()
{
return GetZero();
}
}
</code>
</example>
</member>
<member name="M:TestNamespace.TestClass.GetGenericValue``1(``0)">
<summary>
The GetGenericValue method.
</summary>
<remarks>
This sample shows how to specify the <see
cref="M:TestNamespace.TestClass.GetGenericValue``1(``0)"/> method as a cref attribute.
</remarks>
</member>
<member name="T:TestNamespace.GenericClass`1">
<summary>
GenericClass.
</summary>
<remarks>
This example shows how to specify the <see cref="T:TestNamespace.GenericClass`1"/> type as a cref
attribute.
</remarks>
</member>
</members> <members>
<member name="T:TestNamespace.TestClass">
<summary>
TestClass contains two cref examples.
</summary>
</member>
<member name="M:TestNamespace.TestClass.GetZero">
<summary>
The GetZero method.
</summary>
<example> This sample shows how to call the <see cref="M:TestNamespace.TestClass.GetZero"/>
method.
<code>
class TestClass
{
static int Main()
{
return GetZero();
}
}
</code>
</example>
</member>
<member name="M:TestNamespace.TestClass.GetGenericValue``1(``0)">
<summary>
The GetGenericValue method.
</summary>
<remarks>
This sample shows how to specify the <see
cref="M:TestNamespace.TestClass.GetGenericValue``1(``0)"/> method as a cref attribute.
</remarks>
</member>
<member name="T:TestNamespace.GenericClass`1">
<summary>
GenericClass.
</summary>
<remarks>
This example shows how to specify the <see cref="T:TestNamespace.GenericClass`1"/> type as a cref
attribute.
</remarks>
</member>
</members>
</doc>
Consulte também
Comentários da documentação XML
Marcas recomendadas para comentários de documentação
<example> (Guia de Programação em C#)
23/10/2019 • 2 minutes to read • Edit Online
Sintaxe
<example>description</example>
Parâmetros
description
Uma descrição do exemplo de código.
Comentários
A marca <example> permite especificar um exemplo de como usar um método ou outro membro da biblioteca.
Normalmente, isso envolve o uso da marca <code>.
Compile com -doc para processar comentários de documentação em um arquivo.
Exemplo
// Save this file as CRefTest.cs
// Compile with: csc CRefTest.cs -doc:Results.xml
namespace TestNamespace
{
/// <summary>
/// TestClass contains several cref examples.
/// </summary>
public class TestClass
{
/// <summary>
/// This sample shows how to specify the <see cref="TestClass"/> constructor as a cref attribute.
/// </summary>
public TestClass()
{ }
/// <summary>
/// This sample shows how to specify the <see cref="TestClass(int)"/> constructor as a cref attribute.
/// </summary>
public TestClass(int value)
{ }
/// <summary>
/// The GetZero method.
/// </summary>
/// <example>
/// This sample shows how to call the <see cref="GetZero"/> method.
/// <code>
/// class TestClass
/// {
/// static int Main()
/// {
/// return GetZero();
/// }
/// }
/// </code>
/// </code>
/// </example>
public static int GetZero()
{
return 0;
}
/// <summary>
/// The GetGenericValue method.
/// </summary>
/// <remarks>
/// This sample shows how to specify the <see cref="GetGenericValue"/> method as a cref attribute.
/// </remarks>
/// <summary>
/// GenericClass.
/// </summary>
/// <remarks>
/// This example shows how to specify the <see cref="GenericClass{T}"/> type as a cref attribute.
/// </remarks>
class GenericClass<T>
{
// Fields and members.
}
class Program
{
static int Main()
{
return TestClass.GetZero();
}
}
}
Consulte também
Guia de Programação em C#
Marcas recomendadas para comentários de documentação
<exception> (Guia de Programação em C#)
23/10/2019 • 2 minutes to read • Edit Online
Sintaxe
<exception cref="member">description</exception>
Parâmetros
cref = " member "
Uma referência a uma exceção que está disponível no ambiente de compilação atual. O compilador verifica se a
exceção apresentada existe e move o member para o nome de elemento canônico no XML de saída. member deve
ser exibido entre aspas duplas (" ").
Confira mais informações sobre como formatar member para fazer referência a um tipo genérico em Como
processar o Arquivo XML.
description
Uma descrição da exceção.
Comentários
A marca <exception> permite que você especifique quais exceções podem ser lançadas. Essa marca pode ser
aplicada às definições de métodos, propriedades, eventos e indexadores.
Compile com -doc para processar comentários de documentação em um arquivo.
Para obter mais informações sobre o tratamento de exceção, consulte Exceções e tratamento de exceção.
Exemplo
// compile with: -doc:DocFileName.xml
Sintaxe
<include file='filename' path='tagpath[@name="id"]' />
Parâmetros
filename
O nome do arquivo XML que contém a documentação. O nome do arquivo pode ser qualificado com um caminho
relativo ao arquivo de código-fonte. Coloque filename entre aspas simples (' ').
tagpath
O caminho das marcas em filename que leva à marca name . Coloque o caminho entre aspas simples (' ').
name
O especificador de nome na marca que precede os comentários; name terá um id .
id
A ID da marca que precede os comentários. Coloque a ID entre aspas duplas (" ").
Comentários
A marca <include> permite consultar comentários em outro arquivo que descrevem os tipos e membros em seu
código-fonte. Essa é uma alternativa para inserir comentários de documentação diretamente em seu arquivo de
código-fonte. Colocando a documentação em um arquivo separado, é possível aplicar o controle do código-fonte à
documentação separadamente do código-fonte. Uma pessoa pode fazer o check-out do arquivo de código-fonte e
outra pessoa pode fazer o check-out do arquivo de documentação.
A marca <include> usa a sintaxe XML XPath. Consulte a documentação do XPath para obter maneiras de
personalizar o uso de <include>.
Exemplo
Este é um exemplo de vários arquivos. O primeiro arquivo, que usa <include>, é listado abaixo:
// compile with: -doc:DocFileName.xml
<MyDocs>
<MyMembers name="test">
<summary>
The summary for this type.
</summary>
</MyMembers>
<MyMembers name="test2">
<summary>
The summary for this other type.
</summary>
</MyMembers>
</MyDocs>
Saída do Programa
A seguinte saída é gerada quando você compila as classes Test e Test2 com a seguinte linha de comando:
/doc:DocFileName.xml. No Visual Studio, você especifica a opção de comentários do documento XML no painel
Compilar do Designer de Projeto. Quando o compilador C# encontrar a marca <include>, ele pesquisará os
comentários da documentação em xml_include_tag.doc, em vez de no arquivo de origem atual. Em seguida, o
compilador gera DocFileName.xml e esse é o arquivo consumido pelas ferramentas de documentação como o
DocFX e o Sandcastle para produzir a documentação final.
<?xml version="1.0"?>
<doc>
<assembly>
<name>xml_include_tag</name>
</assembly>
<members>
<member name="T:Test">
<summary>
The summary for this type.
</summary>
</member>
<member name="T:Test2">
<summary>
The summary for this other type.
</summary>
</member>
</members>
</doc>
Consulte também
Guia de Programação em C#
Marcas recomendadas para comentários de documentação
<list> (Guia de Programação em C#)
23/10/2019 • 2 minutes to read • Edit Online
Sintaxe
<list type="bullet" | "number" | "table">
<listheader>
<term>term</term>
<description>description</description>
</listheader>
<item>
<term>term</term>
<description>description</description>
</item>
</list>
Parâmetros
term
Um termo a se definir, que será definido em description .
description
Um item em uma lista com marcadores ou numerada ou uma definição de um term .
Comentários
O bloco <listheader> é usado para definir a linha de cabeçalho de uma tabela ou lista de definição. Ao definir uma
tabela, é necessário fornecer uma entrada para o termo no título.
Cada item na lista é especificado com um bloco <item>. Ao criar uma lista de definições, é necessário especificar
term e description . No entanto, para uma tabela, lista com marcadores ou lista numerada, será necessário
fornecer apenas uma entrada para description .
Uma lista ou tabela pode ter quantos blocos <item> forem necessários.
Compile com -doc para processar comentários de documentação em um arquivo.
Exemplo
// compile with: -doc:DocFileName.xml
Consulte também
Guia de Programação em C#
Marcas recomendadas para comentários de documentação
<para> (Guia de Programação em C#)
23/10/2019 • 2 minutes to read • Edit Online
Sintaxe
<para>content</para>
Parâmetros
content
O texto do parágrafo.
Comentários
A marca <para> é para uso dentro de uma marca, como <summary>, <remarks> ou <returns> e permite que
você adicione estrutura ao texto.
Compile com -doc para processar comentários de documentação em um arquivo.
Exemplo
Consulte <summary> para obter um exemplo sobre o uso de <para>.
Consulte também
Guia de Programação em C#
Marcas recomendadas para comentários de documentação
<param> (Guia de Programação em C#)
23/10/2019 • 2 minutes to read • Edit Online
Sintaxe
<param name="name">description</param>
Parâmetros
name
O nome do parâmetro de um método. Coloque o nome entre aspas duplas (" ").
description
Uma descrição do parâmetro.
Comentários
A marca <param> deve ser usada no comentário para uma declaração de método para descrever um dos
parâmetros do método. Para documentar vários parâmetros, use várias marcas <param>.
O texto da marca <param> será exibido no IntelliSense, o Pesquisador de Objetos e no relatório Web de
comentários sobre código.
Compile com -doc para processar comentários de documentação em um arquivo.
Exemplo
// compile with: -doc:DocFileName.xml
// Multiple parameters.
/// <param name="Int1">Used to indicate status.</param>
/// <param name="Float1">Used to specify context.</param>
public static void DoWork(int Int1, float Float1)
{
}
Consulte também
Guia de Programação em C#
Marcas recomendadas para comentários de documentação
<paramref> (Guia de Programação em C#)
23/10/2019 • 2 minutes to read • Edit Online
Sintaxe
<paramref name="name"/>
Parâmetros
name
O nome do parâmetro ao qual você deseja se referir. Coloque o nome entre aspas duplas (" ").
Comentários
A marca <paramref> fornece uma maneira de indicar que uma palavra nos comentários do código, por exemplo,
em um blogo <summary> ou <remarks> refere-se a um parâmetro. O arquivo XML pode ser processado para
formatar essa palavra de alguma forma distinta, como com uma fonte em negrito ou itálico.
Compile com -doc para processar comentários de documentação em um arquivo.
Exemplo
// compile with: -doc:DocFileName.xml
Consulte também
Guia de Programação em C#
Marcas recomendadas para comentários de documentação
<permission> (Guia de Programação em C#)
23/10/2019 • 2 minutes to read • Edit Online
Sintaxe
<permission cref="member">description</permission>
Parâmetros
cref = " member "
Uma referência a um membro ou campo disponível para ser chamado do ambiente de compilação atual. O
compilador verifica se o elemento de código fornecido existe e converte member no nome de elemento canônico
no XML de saída. member deve ser exibido entre aspas duplas (" ").
Para obter informações sobre como criar uma referência cref para um tipo genérico, consulte <consulte>.
description
Uma descrição do acesso ao membro.
Comentários
A marca <permissão > permite documentar o acesso de um membro. A classe PermissionSet permite que você
especifique o acesso a um membro.
Compile com -doc para processar comentários de documentação em um arquivo.
Exemplo
// compile with: -doc:DocFileName.xml
class TestClass
{
/// <permission cref="System.Security.PermissionSet">Everyone can access this method.</permission>
public static void Test()
{
}
Consulte também
Guia de Programação em C#
Marcas recomendadas para comentários de documentação
<remarks> (Guia de Programação em C#)
23/10/2019 • 2 minutes to read • Edit Online
Sintaxe
<remarks>description</remarks>
Parâmetros
Description
Uma descrição do membro.
Comentários
A marca <remarks> é usada para adicionar informações sobre o tipo, complementando as informações
especificadas com <summary>. Essas informações são exibidas na janela do Pesquisador de Objetos.
Compile com -doc para processar comentários de documentação em um arquivo.
Exemplo
// compile with: -doc:DocFileName.xml
/// <summary>
/// You may have some primary information about this class.
/// </summary>
/// <remarks>
/// You may have some additional information about this class.
/// </remarks>
public class TestClass
{
/// text for Main
static void Main()
{
}
}
Consulte também
Guia de Programação em C#
Marcas recomendadas para comentários de documentação
<returns> (Guia de Programação em C#)
23/10/2019 • 2 minutes to read • Edit Online
Sintaxe
<returns>description</returns>
Parâmetros
description
Uma descrição do valor retornado.
Comentários
A marca <returns> deve ser usada no comentário para uma declaração de método descrever o valor retornado.
Compile com -doc para processar comentários de documentação em um arquivo.
Exemplo
// compile with: -doc:DocFileName.xml
Consulte também
Guia de Programação em C#
Marcas recomendadas para comentários de documentação
<see> (Guia de Programação em C#)
23/10/2019 • 2 minutes to read • Edit Online
Sintaxe
<see cref="member"/>
Parâmetros
cref = " member "
Uma referência a um membro ou campo disponível para ser chamado do ambiente de compilação atual. O
compilador verifica se o elemento de código fornecido existe e passa member para o nome de elemento no XML
de saída. Coloque member entre aspas duplas (“ ”).
Comentários
Use a marca <see> para especificar um link de dentro do texto. Use <seealso> para indicar que o texto deve ser
colocado em uma seção Consulte também. Use o atributo cref para criar hyperlinks internos para páginas de
documentação para elementos de código.
Compile com -doc para processar comentários de documentação em um arquivo.
O exemplo a seguir mostra uma marca <see> dentro de uma seção de resumo.
Consulte também
Guia de Programação em C#
Marcas recomendadas para comentários de documentação
<seealso> (Guia de Programação em C#)
23/10/2019 • 2 minutes to read • Edit Online
Sintaxe
<seealso cref="member"/>
Parâmetros
cref = " member "
Uma referência a um membro ou campo disponível para ser chamado do ambiente de compilação atual. O
compilador verifica se o elemento de código fornecido existe e passa member para o nome de elemento no XML
de saída. member deve ser exibido entre aspas duplas (" ").
Para obter informações sobre como criar uma referência cref para um tipo genérico, consulte <consulte>.
Comentários
A marca <seealso > permite especificar o texto que você quer que seja exibido na seção Ver também. Use <see>
para especificar um link de dentro do texto.
Compile com -doc para processar comentários de documentação em um arquivo.
Exemplo
Consulte <summary> para obter um exemplo sobre o uso de <seealso>.
Consulte também
Guia de Programação em C#
Marcas recomendadas para comentários de documentação
<summary> (Guia de Programação em C#)
23/10/2019 • 2 minutes to read • Edit Online
Sintaxe
<summary>description</summary>
Parâmetros
description
Um resumo do objeto.
Comentários
A marca <summary> deve ser usada para descrever um tipo ou um membro de tipo. Use <remarks> para
adicionar mais informações a uma descrição de tipo. Use o atributo cref para habilitar ferramentas de
documentação como o DocFX e o Sandcastle para criar hiperlinks internos para páginas de documentação em
elementos de código.
O texto da marca <summary> é a única fonte de informações sobre o tipo no IntelliSense e também é exibido
na janela Pesquisador de Objetos.
Compile com -doc para processar comentários de documentação em um arquivo. Para criar a documentação
final com base no arquivo gerado pelo compilador, crie uma ferramenta personalizada ou use uma ferramenta
como o DocFX ou o Sandcastle.
Exemplo
// compile with: -doc:DocFileName.xml
Exemplo
O exemplo a seguir mostra como fazer uma referência cref para um tipo genérico.
// the following cref shows how to specify the reference, such that,
// the compiler will resolve the reference
/// <summary cref="C{T}">
/// </summary>
class A { }
Consulte também
Guia de Programação em C#
Marcas recomendadas para comentários de documentação
<typeparam> (Guia de Programação em C#)
23/10/2019 • 2 minutes to read • Edit Online
Sintaxe
<typeparam name="name">description</typeparam>
Parâmetros
name
O nome do parâmetro de tipo. Coloque o nome entre aspas duplas (" ").
description
Uma descrição do parâmetro de tipo.
Comentários
A marca <typeparam> deve ser usada no comentário para um tipo genérico ou para uma declaração de método
descrever um dos parâmetros de tipo. Adicione uma marca para cada parâmetro de tipo do tipo ou do método
genérico.
Para obter mais informações, consulte Genéricos.
O texto da marca <typeparam> será exibido no IntelliSense, o relatório Web de comentários sobre código da
Janela do Pesquisador de Objetos.
Compile com -doc para processar comentários de documentação em um arquivo.
Exemplo
// compile with: -doc:DocFileName.xml
Consulte também
Referência de C#
Guia de Programação em C#
Marcas recomendadas para comentários de documentação
<typeparamref> (Guia de Programação em C#)
23/10/2019 • 2 minutes to read • Edit Online
Sintaxe
<typeparamref name="name"/>
Parâmetros
name
O nome do parâmetro de tipo. Coloque o nome entre aspas duplas (" ").
Comentários
Para obter mais informações sobre parâmetros de tipo em tipos e métodos genéricos, consulte Genéricos.
Use essa marca para habilitar os consumidores do arquivo de documentação a formatar a palavra de alguma
forma distinta, por exemplo, em itálico.
Compile com -doc para processar comentários de documentação em um arquivo.
Exemplo
// compile with: -doc:DocFileName.xml
Consulte também
Guia de Programação em C#
Marcas recomendadas para comentários de documentação
<value> (Guia de programação em C#)
23/10/2019 • 2 minutes to read • Edit Online
Sintaxe
<value>property-description</value>
Parâmetros
property-description
Uma descrição da propriedade.
Comentários
A marca <value> permite descrever o valor que uma propriedade representa. Observe que quando você adiciona
uma propriedade por meio do assistente de código no ambiente de desenvolvimento do Visual Studio .NET, ele
adicionará uma marca <summary> para a nova propriedade. Então, você deve adicionar manualmente uma marca
<value> para descrever o valor que a propriedade representa.
Compile com -doc para processar comentários de documentação em um arquivo.
Exemplo
// compile with: -doc:DocFileName.xml
Os recursos de manipulação de exceção da linguagem C# ajudam você a lidar com quaisquer situações
excepcionais ou inesperadas que ocorram quando um programa for executado. A manipulação de exceção usa as
palavras-chave try , catch e finally para executar ações que podem não ser bem-sucedidas, lidar com falhas
quando decidir se é razoável fazer isso e limpar recursos posteriormente. As exceções podem ser geradas pelo
CLR (Common Language Runtime), pelo .NET Framework ou por quaisquer bibliotecas de terceiros, ou pelo
código do aplicativo. As exceções são criadas usando a palavra-chave throw .
Em muitos casos, uma exceção pode ser lançada não por um método que seu código chamou diretamente, mas
por outro método mais abaixo na pilha de chamadas. Quando isso acontecer, o CLR desenrolará a pilha,
procurando por um método com um bloco catch para o tipo de exceção específico e ele será executado primeiro
o bloco catch que encontrar. Se ele não encontrar um bloco catch apropriado na pilha de chamadas, ele
encerrará o processo e exibirá uma mensagem para o usuário.
Neste exemplo, um método testa a divisão por zero e captura o erro. Sem a manipulação de exceção, esse
programa encerraria com um DivideByZeroException não resolvido.
class ExceptionTest
{
static double SafeDivision(double x, double y)
{
if (y == 0)
throw new System.DivideByZeroException();
return x / y;
}
static void Main()
{
// Input for test purposes. Change the values to see
// exception handling behavior.
double a = 98, b = 0;
double result = 0;
try
{
result = SafeDivision(a, b);
Console.WriteLine("{0} divided by {1} = {2}", a, b, result);
}
catch (DivideByZeroException e)
{
Console.WriteLine("Attempted divide by zero.");
}
}
}
Seções relacionadas
Consulte os artigos a seguir para obter mais informações sobre exceções e manipulação de exceção:
Usando exceções
Tratamento de Exceção
Criando e lançando exceções
Exceções geradas pelo compilador
Como manipular uma exceção usando try/catch (Guia de programação em C#)
Como executar código de limpeza usando finally
Como capturar uma exceção não compatível com CLS
Especificação da Linguagem C#
Para obter mais informações, veja Exceções na Especificação da linguagem C#. A especificação da linguagem é a
fonte definitiva para a sintaxe e o uso de C#.
Consulte também
SystemException
Guia de Programação em C#
Palavras-chave do C#
throw
try-catch
try-finally
try-catch-finally
Exceções
Usando exceções (Guia de Programação em C#)
31/10/2019 • 6 minutes to read • Edit Online
No C#, os erros no programa em tempo de execução são propagados pelo programa usando um mecanismo
chamado exceções. As exceções são geradas pelo código que encontra um erro e capturadas pelo código que pode
corrigir o erro. As exceções podem ser geradas pelo CLR (Common Language Runtime) do .NET Framework ou
pelo código em um programa. Uma vez que uma exceção é gerada, ela é propagada acima na pilha de chamadas
até uma instrução catch para a exceção ser encontrada. As exceções não capturadas são tratadas por um
manipulador de exceção genérico fornecido pelo sistema que exibe uma caixa de diálogo.
As exceções são representadas por classes derivadas de Exception. Essa classe identifica o tipo de exceção e
contém propriedades que têm detalhes sobre a exceção. Gerar uma exceção envolve criar uma instância de uma
classe derivada de exceção, opcionalmente configurar propriedades da exceção e, em seguida, gerar o objeto
usando a palavra-chave throw . Por exemplo:
}
private static void TestThrow()
{
CustomException ex =
new CustomException("Custom exception in TestThrow()");
throw ex;
}
Depois que uma exceção é gerada, o tempo de execução verifica a instrução atual para ver se ela está dentro de um
bloco try . Se estiver, todos os blocos catch associados ao bloco try serão verificados para ver se eles podem
capturar a exceção. Os blocos Catch normalmente especificam os tipos de exceção. Se o tipo do bloco catch for
do mesmo tipo que a exceção ou uma classe base da exceção, o bloco catch poderá manipular o método. Por
exemplo:
Se a instrução que gera uma exceção não estiver dentro de um bloco try ou se o bloco try que o contém não
tiver um bloco catch correspondente, o tempo de execução verificará o método de chamada quanto a uma
instrução try e blocos catch . O tempo de execução continuará acima na pilha de chamada, pesquisando um
bloco catch compatível. Depois que o bloco catch for localizado e executado, o controle será passado para a
próxima instrução após aquele bloco catch .
Uma instrução try pode conter mais de um bloco catch . A primeira instrução catch que pode manipular a
exceção é executado, todas as instruções catch posteriores, mesmo se forem compatíveis, são ignoradas.
Portanto, os blocos de captura devem sempre ser ordenados do mais específico (ou mais derivado) para o menos
específico. Por exemplo:
using System;
using System.IO;
Console.WriteLine("Done");
}
}
Antes de o bloco catch ser executado, o tempo de execução verifica se há blocos finally . Os blocos Finally
permitem que o programador limpe qualquer estado ambíguo que pode ser deixado de um bloco try cancelado
ou libere quaisquer recursos externos (como identificadores de gráfico, conexões de banco de dados ou fluxos de
arquivo) sem esperar o coletor de lixo no tempo de execução finalizar os objetos. Por exemplo:
static void TestFinally()
{
System.IO.FileStream file = null;
//Change the path to something that works on your machine.
System.IO.FileInfo fileInfo = new System.IO.FileInfo(@"C:\file.txt");
try
{
file = fileInfo.OpenWrite();
file.WriteByte(0xF);
}
finally
{
// Closing the file allows you to reopen it immediately - otherwise IOException is thrown.
if (file != null)
{
file.Close();
}
}
try
{
file = fileInfo.OpenWrite();
System.Console.WriteLine("OpenWrite() succeeded");
}
catch (System.IO.IOException)
{
System.Console.WriteLine("OpenWrite() failed");
}
}
Se WriteByte()gerou uma exceção, o código no segundo bloco try que tentar reabrir o arquivo falhará se
file.Close() não for chamado e o arquivo permanecerá bloqueado. Como os blocos finally são executados
mesmo se uma exceção for gerada, o bloco finally no exemplo anterior permite que o arquivo seja fechado
corretamente e ajuda a evitar um erro.
Se não for encontrado nenhum bloco catch compatível na pilha de chamadas após uma exceção ser gerada,
ocorrerá uma das três coisas:
Se a exceção estiver em um finalizador, o finalizador será anulado e o finalizador base, se houver, será
chamado.
Se a pilha de chamadas contiver um construtor estático ou um inicializador de campo estático, uma
TypeInitializationException será gerada, com a exceção original atribuída à propriedade InnerException da
nova exceção.
Se o início do thread for atingido, o thread será encerrado.
Consulte também
Guia de Programação em C#
Exceções e manipulação de exceções
Manipulação de exceções (Guia de Programação em
C#)
04/11/2019 • 6 minutes to read • Edit Online
Um bloco try é usado por programadores de C# para particionar o código que pode ser afetado por uma exceção.
Os blocos catch associados são usados para tratar qualquer exceção resultante. Um bloco finally contém código
que será executado independentemente de uma exceção ser ou não ser lançada no bloco try , como a liberação
de recursos que estão alocados no bloco try . Um bloco try exige um ou mais blocos catch associados ou um
bloco finally ou ambos.
Os exemplos a seguir mostram uma instrução try-catch , uma instrução try-finally e um instrução
try-catch-finally .
try
{
// Code to try goes here.
}
catch (SomeSpecificException ex)
{
// Code to handle the exception goes here.
// Only catch exceptions that you know how to handle.
// Never catch base class System.Exception without
// rethrowing it at the end of the catch block.
}
try
{
// Code to try goes here.
}
finally
{
// Code to execute after the try block goes here.
}
try
{
// Code to try goes here.
}
catch (SomeSpecificException ex)
{
// Code to handle the exception goes here.
}
finally
{
// Code to execute after the try (and possibly catch) blocks
// goes here.
}
Blocos catch
Um bloco catch pode especificar o tipo de exceção a ser capturado. A especificação de tipo é chamada de filtro
de exceção. O tipo de exceção deve ser derivado de Exception. Em geral, não especifique Exception como o filtro
de exceção, a menos que você saiba como tratar todas as exceções que podem ser lançadas no bloco try ou
incluiu uma instrução throw no final do seu bloco catch .
Vários blocos catch com filtros de exceção diferentes podem ser encadeados. Os blocos catch são avaliados de
cima para baixo no seu código, mas somente um bloco catch será executado para cada exceção que é lançada. O
primeiro bloco catch que especifica o tipo exato ou uma classe base da exceção lançada será executado. Se
nenhum bloco catch especificar um filtro de exceção correspondente, um bloco catch que não tem um filtro
será selecionado, caso haja algum na instrução. É importante posicionar os blocos catch com os tipos de exceção
mais específicos (ou seja, os mais derivados) em primeiro.
Você deve capturar exceções quando as seguintes condições forem verdadeiras:
Você tem uma boa compreensão de porque a exceção seria lançada e pode implementar uma recuperação
específica, como solicitar que o usuário insira um novo nome de arquivo, quando você capturar um objeto
FileNotFoundException.
Você pode criar e lançar uma exceção nova e mais específica.
Você deseja tratar parcialmente uma exceção antes de passá-la para tratamento adicional. No exemplo a
seguir, um bloco catch é usado para adicionar uma entrada a um log de erros antes de lançar novamente
a exceção.
try
{
// Try to access a resource.
}
catch (System.UnauthorizedAccessException e)
{
// Call a custom error logging procedure.
LogError(e);
// Re-throw the error.
throw;
}
Blocos Finally
Um bloco finally permite que você limpe as ações que são realizadas em um bloco try . Se estiver presente, o
bloco finally será executado por último, depois do bloco try e de qualquer bloco catch de correspondência.
Um bloco finally é sempre executado, independentemente de uma exceção ser lançada ou de um bloco catch
correspondente ao tipo de exceção ser encontrado.
O bloco finally pode ser usado para liberar recursos, como fluxos de arquivos, conexões de banco de dados e
identificadores de gráficos, sem esperar que o coletor de lixo no tempo de execução finalize os objetos. Consulte a
Instrução using para obter mais informações.
No exemplo a seguir, o bloco finally é usado para fechar um arquivo está aberto no bloco try . Observe que o
estado do identificador de arquivo é selecionado antes do arquivo ser fechado. Se o bloco try não puder abrir o
arquivo, o identificador de arquivo ainda terá o valor null e o bloco finally não tentará fechá-lo. De outra
forma, se o arquivo for aberto com êxito no bloco try , o bloco finally fechará o arquivo aberto.
Especificação da Linguagem C#
Para obter mais informações, veja Exceções e A declaração try na Especificação da Linguagem C#. A especificação
da linguagem é a fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
Exceções e manipulação de exceções
try-catch
try-finally
try-catch-finally
Instrução using
Criando e lançando exceções (Guia de Programação
em C#)
04/11/2019 • 7 minutes to read • Edit Online
As exceções são usadas para indicar que ocorreu um erro durante a execução do programa. Objetos de exceção
que descrevem um erro são criados e, em seguida, lançados com a palavra-chave throw. Então, o tempo de
execução procura o manipulador de exceção mais compatível.
Os programadores devem lançar exceções quando uma ou mais das seguintes condições forem verdadeiras:
O método não pode concluir sua funcionalidade definida.
Por exemplo, se um parâmetro para um método tem um valor inválido:
class ProgramLog
{
System.IO.FileStream logFile = null;
void OpenLog(System.IO.FileInfo fileName, System.IO.FileMode mode) {}
void WriteLog()
{
if (!this.logFile.CanWrite)
{
throw new System.InvalidOperationException("Logfile cannot be read-only");
}
// Else write data to the log and return.
}
}
As exceções contêm uma propriedade chamada StackTrace. Essa cadeia de caracteres contém o nome dos métodos
na pilha de chamadas atual, junto com o nome de arquivo e o número de linha em que a exceção foi lançada para
cada método. Um objeto StackTrace é criado automaticamente pelo CLR (Common Language Runtime) no ponto
da instrução throw , de modo que as exceções devem ser lançadas do ponto em que o rastreamento de pilha deve
começar.
Todas as exceções contêm uma propriedade chamada Message. Essa cadeia de caracteres deve ser definida para
explicar o motivo da exceção. Observe que as informações que são sensíveis à segurança não devem ser colocadas
no texto da mensagem. Além Message, ArgumentException contém uma propriedade chamada ParamName que
deve ser definida como o nome do argumento que causou a exceção a ser lançada. No caso de um setter de
propriedade ParamName deve ser definido como value .
Os métodos públicos e protegidos devem lançar exceções sempre que não puderem concluir suas funções
pretendidas. A classe de exceção que é lançada deve ser a exceção mais específica disponível que se adapta às
condições do erro. Essas exceções devem ser documentadas como parte da funcionalidade de classe e as classes
derivadas ou as atualizações da classe original devem manter o mesmo comportamento para compatibilidade com
versões anteriores.
Novas propriedades só devem ser adicionadas à classe de exceção quando os dados que elas fornecem são úteis
para resolver a exceção. Se forem adicionadas novas propriedades à classe de exceção derivada, ToString()
deverá ser substituído para retornar as informações adicionadas.
Especificação da Linguagem C#
Para obter mais informações, veja Exceções e A declaração throw na Especificação da Linguagem C#. A
especificação da linguagem é a fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Guia de Programação em C#
Exceções e manipulação de exceções
Hierarquia de exceções
Tratamento de Exceção
Exceções geradas pelo compilador (Guia de
Programação em C#)
31/10/2019 • 2 minutes to read • Edit Online
Algumas exceções são geradas automaticamente pelo CLR (Common Language Runtime) do .NET Framework
quando as operações básicas falham. Essas exceções e suas condições de erro são listadas na tabela a seguir.
EXCEÇÃO DESCRIÇÃO
Consulte também
Guia de Programação em C#
Exceções e manipulação de exceções
Tratamento de Exceção
try-catch
try-finally
try-catch-finally
Como: manipular uma exceção usando try/catch
(Guia de Programação em C#)
23/10/2019 • 2 minutes to read • Edit Online
A finalidade de um bloco try-catch é capturar e manipular uma exceção gerada pelo código de trabalho. Algumas
exceções podem ser manipuladas em um bloco catch e o problema pode ser resolvido sem que a exceção seja
gerada novamente. No entanto, com mais frequência, a única coisa que você pode fazer é certificar-se de que a
exceção apropriada seja gerada.
Exemplo
Neste exemplo, IndexOutOfRangeException não é a exceção mais apropriada: ArgumentOutOfRangeException
faz mais sentido para o método porque o erro é causado pelo argumento index passado pelo chamador.
class TestTryCatch
{
static int GetInt(int[] array, int index)
{
try
{
return array[index];
}
catch (System.IndexOutOfRangeException e) // CS0168
{
System.Console.WriteLine(e.Message);
// Set IndexOutOfRangeException to the new exception's InnerException.
throw new System.ArgumentOutOfRangeException("index parameter is out of range.", e);
}
}
}
Comentários
O código que causa uma exceção fica dentro do bloco try . A instrução catch é adicionada logo após para
manipular IndexOutOfRangeException , se ocorrer. O bloco catch manipula o IndexOutOfRangeException e gera a
exceção ArgumentOutOfRangeException , que é mais adequada. Para fornecer ao chamador tantas informações
quanto possível, considere especificar a exceção original como o InnerException da nova exceção. Uma vez que a
propriedade InnerException é readonly, você precisa atribuí-la no construtor da nova exceção.
Consulte também
Guia de Programação em C#
Exceções e manipulação de exceções
Tratamento de Exceção
Como: executar código de limpeza usando finally
(Guia de Programação em C#)
23/10/2019 • 2 minutes to read • Edit Online
O propósito de uma instrução finally é garantir que a limpeza necessária de objetos, normalmente objetos que
estão mantendo recursos externos, ocorra imediatamente, mesmo que uma exceção seja lançada. Um exemplo
dessa limpeza é chamar Close em um FileStream imediatamente após o uso, em vez de esperar que o objeto
passe pela coleta de lixo feita pelo Common Language Runtime, da seguinte maneira:
file = fileInfo.OpenWrite();
file.WriteByte(0xF);
file.Close();
}
Exemplo
Para transformar o código anterior em uma instrução try-catch-finally , o código de limpeza é separado do
código funcional, da seguinte maneira.
try
{
fileInfo = new System.IO.FileInfo("C:\\file.txt");
file = fileInfo.OpenWrite();
file.WriteByte(0xF);
}
catch(System.UnauthorizedAccessException e)
{
System.Console.WriteLine(e.Message);
}
finally
{
if (file != null)
{
file.Close();
}
}
}
Como uma exceção pode ocorrer a qualquer momento no bloco try antes da chamada OpenWrite() ou a
própria chamada OpenWrite() poderia falhar, não há garantia de que o arquivo esteja aberto ao tentarmos fechá-
lo. O bloco finally adiciona uma verificação para verificar se o objeto FileStream não é null antes de chamar o
método Close. Sem a verificação de null , o bloco finally poderia lançar sua própria NullReferenceException,
mas o lançamento de exceções em blocos finally deve ser evitado, se possível.
Uma conexão de banco de dados é outra boa candidata a ser fechada em um bloco finally . Como o número de
conexões permitidas para um servidor de banco de dados é, às vezes, limitado, você deve fechar conexões de
banco de dados assim que possível. Se uma exceção for lançada antes de fechar a conexão, este seria outro caso
em que o uso do bloco finally é melhor que esperar pela coleta de lixo.
Consulte também
Guia de Programação em C#
Exceções e manipulação de exceções
Tratamento de Exceção
Instrução using
try-catch
try-finally
try-catch-finally
Como: Capturar uma exceção não CLS
23/10/2019 • 2 minutes to read • Edit Online
Algumas linguagens .NET, incluindo o C++/CLI, permite que os objetos lancem exceções que não derivam de
Exception. Essas exceções são chamadas de exceções não CLS ou não exceções. Em C#, não é possível gerar
exceções que não sejam do CLS, mas você pode capturá-las de duas maneiras:
Em um bloco catch (RuntimeWrappedException e) .
Por padrão, um assembly do Visual C# captura exceções não CLS como exceções encapsuladas. Use este
método se você precisar de acesso à exceção original, que pode ser acessada por meio da propriedade
RuntimeWrappedException.WrappedException. O procedimento, mais adiante neste tópico, explica como
capturar exceções dessa maneira.
Em um bloco de captura geral (um bloco de captura sem um tipo de exceção especificado), que é colocado
após todos os outros blocos catch .
Use esse método quando desejar realizar alguma ação (como gravar em um arquivo de log) em resposta a
exceções não CLS e você não precisa de acesso às informações de exceção. Por padrão, o Common
Language Runtime encapsula todas as exceções. Para desabilitar esse comportamento, adicione esse
atributo de nível de assembly em seu código, geralmente no arquivo AssemblyInfo.cs:
[assembly: RuntimeCompatibilityAttribute(WrapNonExceptionThrows = false)] .
Exemplo
O exemplo a seguir mostra como capturar uma exceção que não é do CLS, que foi gerada por uma biblioteca de
classes escrita em C++/CLI. Observe que, neste exemplo, o código cliente C# sabe com antecedência que o tipo
da exceção que está sendo gerada é um System.String. Você pode converter a propriedade
RuntimeWrappedException.WrappedException de volta a seu tipo original, desde que o tipo seja acessível por
meio do código.
try
{
// throws gcnew System::String(
// "I do not derive from System.Exception!");
myClass.TestThrow();
}
catch (RuntimeWrappedException e)
{
String s = e.WrappedException as String;
if (s != null)
{
Console.WriteLine(s);
}
}
Consulte também
RuntimeWrappedException
Exceções e manipulação de exceções
Sistema de arquivos e o Registro (Guia de
Programação em C#)
23/10/2019 • 2 minutes to read • Edit Online
Os tópicos a seguir mostram como usar o C# e o .NET Framework para executar várias operações básicas em
arquivos, pastas e no Registro.
Nesta seção
TÍTULO DESCRIÇÃO
Como: iterar em uma árvore de diretório Mostra como iterar manualmente em uma árvore de
diretório.
Como: obter informações sobre arquivos, pastas e unidades Mostra como recuperar informações como tempo de criação
e tamanho, sobre arquivos, pastas e unidades.
Como: criar um arquivo ou uma pasta Mostra como criar um novo arquivo ou pasta.
Como: copiar, excluir e mover arquivos e pastas (Guia de Mostra como copiar, excluir e mover arquivos e pastas.
Programação em C#)
Como: fornecer uma caixa de diálogo de progresso para Mostra como exibir uma caixa de diálogo de progresso
operações de arquivo padrão do Windows para determinadas operações de
arquivo.
Como: ler um arquivo de texto uma linha de cada vez Mostra como recuperar o texto do arquivo uma linha por vez.
Como: criar uma chave no Registro Mostra como gravar uma chave no Registro do sistema.
Seções relacionadas
E/S de arquivo e de fluxo
Como: copiar, excluir e mover arquivos e pastas (Guia de Programação em C#)
Guia de Programação em C#
Arquivos, pastas e unidades
System.IO
Como: iterar em uma árvore de diretório (Guia de
Programação em C#)
23/10/2019 • 10 minutes to read • Edit Online
A expressão "iterar uma árvore de diretório" significa acessar cada arquivo em cada subdiretório aninhado em
uma pasta raiz especificada, em qualquer profundidade. Você não precisa necessariamente abrir cada arquivo.
Você pode recuperar apenas o nome do arquivo ou subdiretório como um string , ou então você pode recuperar
informações adicionais na forma de um objeto System.IO.FileInfo ou System.IO.DirectoryInfo.
NOTE
No Windows, os termos "diretório" e "pasta" são usados de forma intercambiável. A maior parte do texto da documentação e
da interface do usuário usa o termo "pasta", mas a biblioteca de classes do .NET Framework usa o termo "diretório".
No caso mais simples, em que você sabe com certeza que tem permissões de acesso a todos os diretórios em uma
raiz especificada, é possível usar o sinalizador System.IO.SearchOption.AllDirectories . Esse sinalizador retorna
todos os subdiretórios aninhados que correspondem ao padrão especificado. O exemplo a seguir mostra como
usar o sinalizador.
root.GetDirectories("*.*", System.IO.SearchOption.AllDirectories);
As desvantagem dessa abordagem é que, se qualquer um dos subdiretórios na raiz especificada causar um
DirectoryNotFoundException ou UnauthorizedAccessException, o método inteiro falhará e não retornará nenhum
diretório. O mesmo é verdadeiro quando você usa o método GetFiles. Se precisar manipular essas exceções em
subpastas específicas, você precisa percorrer manualmente a árvore de diretório, conforme mostrado nos
exemplos a seguir.
Quando percorre manualmente uma árvore de diretório, você pode manipular os subdiretórios primeiro
(passagem da pré-encomenda) ou os arquivos primeiro (passagem da pós-encomenda). Se executar uma
passagem da pré-encomenda, você percorre toda a árvore na pasta atual antes iterar nos arquivos que estão
diretamente na própria pasta. Os exemplos mais adiante neste documento executam a passagem da pós-
encomenda, mas você pode facilmente modificá-los para executar a passagem da pré-encomenda.
Outra opção é usar a recursão ou uma passagem baseada em pilha. Os exemplos mais adiante neste documento
mostram as duas abordagens.
Se precisar executar uma série de operações em arquivos e pastas, você pode modularizar esses exemplos
refatorando a operação em funções separadas que podem ser invocadas usando um único delegado.
NOTE
Sistemas de arquivos NTFS podem conter pontos de nova análise na forma de pontos de junção, links simbólicos e links
físicos. Os métodos do .NET Framework, tais como GetFiles e GetDirectories, não retornarão nenhum subdiretório em um
ponto de nova análise. Esse comportamento protege contra o risco de entrar em um loop infinito quando dois pontos de
nova análise fazem referência um ao outro. De modo geral, você deve ter muito cuidado ao lidar com pontos de nova análise
para garantir que arquivos não sejam modificados ou excluídos inadvertidamente. Se precisar ter um controle preciso sobre
pontos de nova análise, use a invocação de plataforma ou código nativo para chamar diretamente os métodos apropriados
do sistema de arquivos Win32.
Exemplo
O exemplo a seguir mostra como percorrer uma árvore de diretório usando a recursão. A abordagem recursiva é
elegante, mas tem o potencial de causar uma exceção de estouro de pilha se a árvore de diretório for grande e
profundamente aninhada.
As exceções específicas que são tratadas, bem como as ações específicas que são executadas em cada arquivo ou
pasta, são fornecidas apenas como exemplo. Você deve modificar este código para atender às suas necessidades
específicas. Consulte os comentários no código para obter mais informações.
if (files != null)
{
foreach (System.IO.FileInfo fi in files)
{
// In this example, we only access the existing FileInfo object. If we
// want to open, delete or modify the file, then
// a try-catch block is required here to handle the case
// where the file has been deleted since the call to TraverseTree().
Console.WriteLine(fi.FullName);
}
Exemplo
O exemplo a seguir mostra como iterar em arquivos e pastas em uma árvore de diretório sem o uso de recursão.
Essa técnica usa o tipo de coleção genérico Stack<T>, que é uma pilha UEPS (último a entrar, primeiro a sair).
As exceções específicas que são tratadas, bem como as ações específicas que são executadas em cada arquivo ou
pasta, são fornecidas apenas como exemplo. Você deve modificar este código para atender às suas necessidades
específicas. Consulte os comentários no código para obter mais informações.
if (!System.IO.Directory.Exists(root))
{
throw new ArgumentException();
}
dirs.Push(root);
catch (UnauthorizedAccessException e)
{
Console.WriteLine(e.Message);
continue;
}
catch (System.IO.DirectoryNotFoundException e)
{
Console.WriteLine(e.Message);
continue;
}
// Perform the required action on each file here.
// Modify this block to perform your required task.
foreach (string file in files)
{
try
{
// Perform whatever action is required in your scenario.
System.IO.FileInfo fi = new System.IO.FileInfo(file);
Console.WriteLine("{0}: {1}, {2}", fi.Name, fi.Length, fi.CreationTime);
}
catch (System.IO.FileNotFoundException e)
{
// If file was deleted by a separate application
// or thread since the call to TraverseTree()
// then just continue.
Console.WriteLine(e.Message);
continue;
}
}
Geralmente, é muito demorado testar cada pasta para determinar se seu aplicativo tem permissão para abri-la.
Portanto, o exemplo de código apenas coloca essa parte da operação em um bloco try/catch . É possível
modificar o bloco catch para que, quando lhe for negado acesso a uma pasta, você possa tentar elevar as
permissões e acessá-la novamente. Como regra, capture apenas as exceções que você puder manipular sem deixar
seu aplicativo em um estado desconhecido.
Se você precisar armazenar o conteúdo de uma árvore de diretório, seja na memória ou no disco, a melhor opção
é armazenar apenas a propriedade FullName (do tipo string ) para cada arquivo. Você pode, então, usar essa
cadeia de caracteres para criar um novo objeto FileInfo ou DirectoryInfo, conforme necessário ou abra qualquer
arquivo que precisar de processamento adicional.
Programação robusta
O código de iteração de arquivo robusto deve levar em conta muitas complexidades do sistema de arquivos. Para
saber mais sobre o sistema de arquivos do Windows, confira Visão geral do NTFS.
Consulte também
System.IO
LINQ e Diretórios de Arquivos
Sistema de arquivos e o Registro (Guia de Programação em C#)
Como: obter informações sobre arquivos, pastas e
unidades (Guia de Programação em C#)
23/10/2019 • 4 minutes to read • Edit Online
No .NET Framework, você pode acessar informações do sistema de arquivos usando as classes a seguir:
System.IO.FileInfo
System.IO.DirectoryInfo
System.IO.DriveInfo
System.IO.Directory
System.IO.File
As classes FileInfo e DirectoryInfo representam um arquivo ou diretório e contêm propriedades que expõem
muitos dos atributos de arquivo que têm suporte pelo sistema de arquivos NTFS. Elas também contêm métodos
para abrir, fechar, mover e excluir arquivos e pastas. Você pode criar instâncias dessas classes passando uma cadeia
de caracteres que representa o nome do arquivo, pasta ou unidade para o construtor:
Você também pode obter os nomes das unidades, pastas ou arquivos por meio de chamadas para
DirectoryInfo.GetDirectories, DirectoryInfo.GetFiles e DriveInfo.RootDirectory.
As classes System.IO.Directory e System.IO.File fornecem métodos estáticos para recuperar informações sobre
arquivos e diretórios.
Exemplo
O exemplo a seguir mostra várias maneiras de acessar informações sobre arquivos e pastas.
class FileSysInfo
{
static void Main()
{
// You can also use System.Environment.GetLogicalDrives to
// obtain names of all logical drives on the computer.
System.IO.DriveInfo di = new System.IO.DriveInfo(@"C:\");
Console.WriteLine(di.TotalFreeSpace);
Console.WriteLine(di.VolumeLabel);
// Get the root directory and print out some information about it.
System.IO.DirectoryInfo dirInfo = di.RootDirectory;
Console.WriteLine(dirInfo.Attributes.ToString());
// Get the files in the directory and print out some information about them.
System.IO.FileInfo[] fileNames = dirInfo.GetFiles("*.*");
System.IO.Directory.SetCurrentDirectory(@"C:\Users\Public\TestFolder\");
currentDirName = System.IO.Directory.GetCurrentDirectory();
Console.WriteLine(currentDirName);
Programação robusta
Quando você processa cadeias de caracteres do caminho especificado pelo usuário, você também deve tratar
exceções para as seguintes condições:
O nome do arquivo está malformado. Por exemplo, ele contém caracteres inválidos ou somente espaço em
branco.
O nome do arquivo é nulo.
O nome de arquivo é maior que o comprimento máximo definido pelo sistema.
O nome de arquivo contém dois-pontos (:).
Se o aplicativo não tem permissões suficientes para ler o arquivo especificado, o método Exists retorna false
independentemente de se um caminho existe, o método não gera uma exceção.
Consulte também
System.IO
Guia de Programação em C#
Sistema de arquivos e o Registro (Guia de Programação em C#)
Como: criar um arquivo ou uma pasta (Guia de
Programação em C#)
23/10/2019 • 5 minutes to read • Edit Online
Você pode criar uma pasta no seu computador, criar uma subpasta, criar um arquivo na subpasta e gravar dados
no arquivo programaticamente.
Exemplo
public class CreateFileOrFolder
{
static void Main()
{
// Specify a name for your top-level folder.
string folderName = @"c:\Top-Level Folder";
// You can write out the path name directly instead of using the Combine
// method. Combine just makes the process easier.
string pathString2 = @"c:\Top-Level Folder\SubFolder2";
// You can extend the depth of your path if you want to.
//pathString = System.IO.Path.Combine(pathString, "SubSubFolder");
// Create the subfolder. You can verify in File Explorer that you have this
// structure in the C: drive.
// Local Disk (C:)
// Top-Level Folder
// SubFolder
System.IO.Directory.CreateDirectory(pathString);
// This example uses a random string for the name, but you also can specify
// a particular name.
//string fileName = "MyNewFile.txt";
// Check that the file doesn't already exist. If it doesn't exist, create
// the file and write integers 0 - 99 to it.
// DANGER: System.IO.File.Create will overwrite the file if it already exists.
// This could happen even with random file names, although it is unlikely.
if (!System.IO.File.Exists(pathString))
{
using (System.IO.FileStream fs = System.IO.File.Create(pathString))
{
for (byte i = 0; i < 100; i++)
{
fs.WriteByte(i);
}
}
}
}
else
{
Console.WriteLine("File \"{0}\" already exists.", fileName);
return;
}
//0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29
//30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56
// 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 8
//3 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99
}
Se a pasta já existir, CreateDirectory não fará nada, e nenhuma exceção será gerada. No entanto, File.Create
substituirá um arquivo existente por um novo. O exemplo usa uma instrução if - else para impedir que um
arquivo existente seja substituído.
Fazendo as alterações a seguir no exemplo, você pode especificar diferentes resultados com base em se já existe
um arquivo com um determinado nome. Se esse arquivo não existir, o código criará um. Se esse arquivo existir, o
código acrescentará dados a esse arquivo.
Especifique um nome de arquivo não aleatório.
Consulte também
System.IO
Guia de Programação em C#
Sistema de arquivos e o Registro (Guia de Programação em C#)
Como: copiar, excluir e mover arquivos e pastas (Guia
de Programação em C#)
23/10/2019 • 4 minutes to read • Edit Online
Os exemplos a seguir mostram como copiar, mover e excluir arquivos e pastas de maneira síncrona usando as
classes System.IO.File, System.IO.Directory, System.IO.FileInfo e System.IO.DirectoryInfo do namespace
System.IO. Esses exemplos não fornecem uma barra de progresso ou qualquer outra interface do usuário. Se você
quiser fornecer uma caixa de diálogo de progresso padrão, confira Como fornecer uma caixa de diálogo de
progresso para operações de arquivo.
Use System.IO.FileSystemWatcher para fornecer eventos que permitirão que você calcule o progresso ao operar
em vários arquivos. Outra abordagem é usar a invocação de plataforma para invocar os métodos relacionados ao
arquivo relevantes no Shell do Windows. Para obter informações sobre como executar essas operações de arquivo
de forma assíncrona, consulte E/S de arquivo assíncrona.
Exemplo
O exemplo a seguir mostra como copiar arquivos e diretórios.
// Simple synchronous file copy operations with no user interface.
// To run this sample, first create the following directories and files:
// C:\Users\Public\TestFolder
// C:\Users\Public\TestFolder\test.txt
// C:\Users\Public\TestFolder\SubDir\test.txt
public class SimpleFileCopy
{
static void Main()
{
string fileName = "test.txt";
string sourcePath = @"C:\Users\Public\TestFolder";
string targetPath = @"C:\Users\Public\TestFolder\SubDir";
// Copy the files and overwrite destination files if they already exist.
foreach (string s in files)
{
// Use static Path methods to extract only the file name from the path.
fileName = System.IO.Path.GetFileName(s);
destFile = System.IO.Path.Combine(targetPath, fileName);
System.IO.File.Copy(s, destFile, true);
}
}
else
{
Console.WriteLine("Source path does not exist!");
}
Exemplo
O exemplo a seguir mostra como mover arquivos e diretórios.
// Simple synchronous file move operations with no user interface.
public class SimpleFileMove
{
static void Main()
{
string sourceFile = @"C:\Users\Public\public\test.txt";
string destinationFile = @"C:\Users\Public\private\test.txt";
Exemplo
O exemplo a seguir mostra como excluir arquivos e diretórios.
catch (System.IO.IOException e)
{
Console.WriteLine(e.Message);
}
}
}
}
Consulte também
System.IO
Guia de Programação em C#
Sistema de arquivos e o Registro (Guia de Programação em C#)
Como: fornecer uma caixa de diálogo de progresso para operações de arquivo
E/S de arquivo e de fluxo
Tarefas comuns de E/S
Como: fornecer uma caixa de diálogo de progresso
para operações de arquivo (Guia de Programação
em C#)
23/10/2019 • 2 minutes to read • Edit Online
Você pode fornecer uma caixa de diálogo padrão que mostra o andamento em operações de arquivos no
Windows se você usar o método CopyFile(String, String, UIOption) no namespace Microsoft.VisualBasic.
NOTE
Seu computador pode mostrar diferentes nomes ou locais para alguns dos elementos de interface do usuário do Visual
Studio nas instruções a seguir. A edição do Visual Studio que você possui e as configurações que você usa determinam esses
elementos. Para obter mais informações, consulte Personalizando o IDE.
Exemplo
O código a seguir copia o diretório que sourcePath especifica, para o diretório que destinationPath especifica.
Esse código também fornece uma caixa de diálogo padrão que mostra a quantidade estimada de tempo que resta
antes da conclusão da operação.
class FileProgress
{
static void Main()
{
// Specify the path to a folder that you want to copy. If the folder is small,
// you won't have time to see the progress dialog box.
string sourcePath = @"C:\Windows\symbols\";
// Choose a destination for the copied files.
string destinationPath = @"C:\TestFolder";
FileSystem.CopyDirectory(sourcePath, destinationPath,
UIOption.AllDialogs);
}
}
Consulte também
Sistema de arquivos e o Registro (Guia de Programação em C#)
Como: escrever em um arquivo de texto (Guia de
Programação em C#)
23/10/2019 • 3 minutes to read • Edit Online
Estes exemplos mostram várias maneiras de gravar textos em um arquivo. Os dois primeiros exemplos usam
métodos de conveniência estáticos na classe System.IO.File para gravar cada elemento de qualquer
IEnumerable<string> e uma cadeia de caracteres em um arquivo de texto. O exemplo 3 mostra como adicionar
texto a um arquivo quando você precisa processar cada linha individualmente enquanto escreve no arquivo. Os
exemplos de 1 a 3 substituem todo o conteúdo existente no arquivo, mas o exemplo 4 mostra como adicionar
texto em um arquivo existente.
Todos esses exemplos gravam literais de cadeia de caracteres em arquivos. Se você quiser formatar o texto
gravado em um arquivo, use o método Format ou o recurso de interpolação de cadeia de caracteres do C#.
Exemplo
class WriteTextFile
{
static void Main()
{
Programação robusta
As seguintes condições podem causar uma exceção:
O arquivo existe e é somente leitura.
O nome do caminho pode ser muito longo.
O disco pode estar cheio.
Consulte também
Guia de Programação em C#
Sistema de arquivos e o Registro (Guia de Programação em C#)
Exemplo: salvar uma coleção no armazenamento de aplicativos
Como: ler um arquivo de texto (Guia de
Programação em C#)
23/10/2019 • 2 minutes to read • Edit Online
Este exemplo lê o conteúdo de um arquivo de texto usando os métodos estáticos ReadAllText e ReadAllLines da
classe System.IO.File.
Para obter um exemplo que use StreamReader, confira Como ler um arquivo de texto uma linha de cada vez.
NOTE
Os arquivos usados neste exemplo são criados no tópico Como gravar em um arquivo de texto
Exemplo
class ReadFromFile
{
static void Main()
{
// The files used in this example are created in the topic
// How to: Write to a Text File. You can change the path and
// file name to substitute text files of your own.
// Example #1
// Read the file as one string.
string text = System.IO.File.ReadAllText(@"C:\Users\Public\TestFolder\WriteText.txt");
// Example #2
// Read each line of the file into a string array. Each element
// of the array is one line of the file.
string[] lines = System.IO.File.ReadAllLines(@"C:\Users\Public\TestFolder\WriteLines2.txt");
Compilando o código
Copie o código e cole-o em um aplicativo de console em C#.
Se você não estiver usando os arquivos de texto de Como gravar em um arquivo de texto, substitua o argumento
de ReadAllText e ReadAllLines pelo nome de arquivo e pelo caminho adequado em seu computador.
Programação robusta
As seguintes condições podem causar uma exceção:
O arquivo não existe ou não existe no local especificado. Verifique o caminho e a ortografia do nome do
arquivo.
Consulte também
System.IO
Guia de Programação em C#
Sistema de arquivos e o Registro (Guia de Programação em C#)
Como: Ler um arquivo de texto uma linha de cada
vez (Visual C#)
23/10/2019 • 2 minutes to read • Edit Online
Este exemplo lê o conteúdo de um arquivo de texto, uma linha por vez, em uma cadeia de caracteres usando o
método ReadLine da classe StreamReader . Cada linha de texto é armazenada na cadeia de caracteres line e
exibida na tela.
Exemplo
int counter = 0;
string line;
file.Close();
System.Console.WriteLine("There were {0} lines.", counter);
// Suspend the screen.
System.Console.ReadLine();
Compilando o código
Copie o código e cole-o no método Main de um aplicativo de console.
Substitua "c:\test.txt" pelo nome do arquivo real.
Programação robusta
As seguintes condições podem causar uma exceção:
O arquivo pode não existir.
Consulte também
System.IO
Guia de Programação em C#
Sistema de arquivos e o Registro (Guia de Programação em C#)
Como: Criar uma chave no Registro (Visual C#)
23/10/2019 • 3 minutes to read • Edit Online
Este exemplo adiciona o par de valores, "Name" e "Isabella", ao Registro do usuário atual, sob a chave "Names".
Exemplo
Microsoft.Win32.RegistryKey key;
key = Microsoft.Win32.Registry.CurrentUser.CreateSubKey("Names");
key.SetValue("Name", "Isabella");
key.Close();
Compilando o código
Copie o código e cole-o no método Main de um aplicativo de console.
Substitua o parâmetro Names pelo nome de uma chave existente diretamente sob o nó
HKEY_CURRENT_USER do Registro.
Substitua o parâmetro Name pelo nome de um valor que existe diretamente sob o nó Names.
Programação robusta
Analise a estrutura do Registro para encontrar um local adequado para a chave. Por exemplo, caso você queira
abrir a chave Software do usuário atual e criar uma chave com o nome da empresa. Em seguida, adicione os
valores do Registro à chave da empresa.
As seguintes condições podem causar uma exceção:
O nome da chave é nulo.
O usuário não tem permissões para criar chaves do Registro.
O nome da chave excede o limite de 255 caracteres.
A chave é fechada.
A chave do Registro é somente leitura.
Não é seguro armazenar segredos, como senhas, no Registro como texto sem formatação, mesmo se a chave do
Registro estiver protegida por ACL (listas de controle de acesso).
Consulte também
System.IO
Guia de Programação em C#
Sistema de arquivos e o Registro (Guia de Programação em C#)
Ler, gravar e excluir do Registro com C#
Interoperabilidade (Guia de Programação em C#)
04/11/2019 • 2 minutes to read • Edit Online
A interoperabilidade permite que você mantenha e aproveite os investimentos existentes em código não
gerenciado. O código que é executado sob o controle do CLR (Common Language Runtime) é chamado de
código gerenciado, e o código que é executado fora do CLR é chamado de código não gerenciado. COM, COM+,
componentes do C++, componentes do ActiveX e a API do Microsoft Windows são exemplos de código não
gerenciado.
O .NET Framework habilita a interoperabilidade com código não gerenciado por meio de serviços de invocação
de plataforma, o System.Runtime.InteropServices namespace, a interoperabilidade com C++ e a
interoperabilidade COM.
Nesta seção
Visão geral sobre interoperabilidade
Descreve métodos para fins de interoperabilidade entre código gerenciado em C# e código não gerenciado.
Como acessar objetos de interoperabilidade do Office usando recursos do Visual C#
Descreve os recursos que são introduzidos no Visual C# para facilitar a programação do Office.
Como usar propriedades indexadas na programação para interoperabilidade COM
Descreve como usar propriedades indexadas para acesso propriedades COM que têm parâmetros.
Como usar invocação de plataforma para executar um arquivo wave
Descreve como usar os serviços de invocação de plataforma para reproduzir um arquivo de som .wav no sistema
operacional Windows.
Passo a passo: programação do Office
Mostra como criar uma planilha do Excel e um documento do Word com um link para a planilha.
Exemplo de classe COM
Demonstra como expor uma classe C# como um objeto COM.
Especificação da Linguagem C#
Para obter mais informações, veja Noções básicas na Especificação da linguagem C#. A especificação da
linguagem é a fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Marshal.ReleaseComObject
Guia de Programação em C#
Interoperação com código não gerenciado
Passo a passo: programação do Office
Visão geral sobre interoperabilidade (Guia de
Programação em C#)
23/10/2019 • 6 minutes to read • Edit Online
O tópico descreve métodos para permitir a interoperabilidade entre código gerenciado e código não gerenciado
do C#.
Invocação de plataforma
A invocação de plataforma é um serviço que permite ao código gerenciado chamar funções não gerenciadas que
são implementadas em DLLs (bibliotecas de vínculo dinâmico), como aquelas na API do Microsoft Windows. Ela
localiza e invoca uma função exportada e realiza marshaling dos argumentos (inteiros, cadeias de caracteres,
matrizes, estruturas e assim por diante) além do limite de interoperação, conforme necessário.
Para saber mais, veja Consumo de funções de DLL não gerenciadas e Como usar a invocação de plataforma para
reproduzir um arquivo wave.
NOTE
O CLR (Common Language Runtime) gerencia o acesso aos recursos do sistema. Chamar código não gerenciado que esteja
fora do CLR ignora esse mecanismo de segurança e, portanto, apresenta um risco de segurança. Por exemplo, o código não
gerenciado pode chamar recursos diretamente em código não gerenciado, ignorando os mecanismos de segurança do CLR.
Para obter mais informações, confira Segurança no .NET.
Interoperabilidade C++
Você pode usar a interoperabilidade do C++, também conhecida como IJW (It Just Works), para encapsular uma
classe de C++ nativa, de forma que ela possa ser consumida pelo código que é criado no C# ou em outra
linguagem do .NET Framework. Para fazer isso, você deve escrever código C++ para encapsular um componente
nativo DLL ou COM. Ao contrário de outras linguagens do .NET Framework, o Visual C++ tem suporte de
interoperabilidade que permite que o código gerenciado e não gerenciado seja localizado no mesmo aplicativo e
até no mesmo arquivo. Então, você compila o código C++ usando a opção do compilador /clr para produzir um
assembly gerenciado. Finalmente, você adiciona uma referência ao assembly no seu projeto do C# e usa os
objetos encapsulados, assim como usaria outras classes gerenciadas.
Consulte também
Melhorando o desempenho de interoperabilidade
Introdução à interoperabilidade entre COM e .NET
Introdução à interoperabilidade COM em Visual Basic
Marshaling entre código gerenciado e não gerenciado
Interoperação com código não gerenciado
Guia de Programação em C#
Como acessar objetos de interoperabilidade do
Office usando funcionalidades do Visual C# (Guia de
Programação em C#)
04/11/2019 • 19 minutes to read • Edit Online
O Visual C# tem funcionalidades que simplificam o acesso a objetos de API do Office. Os novos recursos
incluem argumentos nomeados e opcionais, um novo tipo chamado dynamic e a capacidade de passar
argumentos para parâmetros de referência em métodos COM como se fossem parâmetros de valor.
Neste tópico, você usará os novos recursos para escrever código que cria e exibe uma planilha do Microsoft
Office Excel. Em seguida, você irá escrever código para adicionar um documento do Office Word que contenha
um ícone que esteja vinculado à planilha do Excel.
Para concluir este passo a passo, você deve ter o Microsoft Office Excel 2007 e o Microsoft Office Word 2007 ou
versões posteriores, instaladas no computador.
NOTE
Seu computador pode mostrar diferentes nomes ou locais para alguns dos elementos de interface do usuário do Visual
Studio nas instruções a seguir. A edição do Visual Studio que você possui e as configurações que você usa determinam
esses elementos. Para obter mais informações, consulte Personalizando o IDE.
2. Adicione o seguinte código ao método Main para criar uma lista bankAccounts que contenha duas contas.
2. Adicione o código a seguir no final de DisplayInExcel . O código insere valores nas duas primeiras colunas
da primeira linha da planilha.
3. Adicione o código a seguir no final de DisplayInExcel . O loop foreach coloca as informações da lista de
contas nas duas primeiras colunas de sucessivas linhas da planilha.
var row = 1;
foreach (var acct in accounts)
{
row++;
workSheet.Cells[row, "A"] = acct.ID;
workSheet.Cells[row, "B"] = acct.Balance;
}
4. Adicione o seguinte código no final de DisplayInExcel para ajustar as larguras das colunas para adequar
o conteúdo.
workSheet.Columns[1].AutoFit();
workSheet.Columns[2].AutoFit();
((Excel.Range)workSheet.Columns[1]).AutoFit();
((Excel.Range)workSheet.Columns[2]).AutoFit();
C#4, e versões posteriores, converte o Object retornado para dynamic automaticamente se o assembly
for referenciado pela opção do compilador -link ou, de maneira equivalente, se a propriedade inserir tipos
de interoperabilidade do Excel estiver definida como true. True é o valor padrão para essa propriedade.
2. Pressione CTRL+F5.
Uma planilha do Excel é exibida contendo os dados das duas contas.
// The Add method has four reference parameters, all of which are
// optional. Visual C# allows you to omit arguments for them if
// the default values are what you want.
wordApp.Documents.Add();
// The Add method has four parameters, all of which are optional.
// In Visual C# 2008 and earlier versions, an argument has to be sent
// for every parameter. Because the parameters are reference
// parameters of type object, you have to create an object variable
// for the arguments that represents 'no value'.
3. Adicione a instrução a seguir no final de DisplayInExcel . O método Copy adiciona a planilha na área de
transferência.
// Put the spreadsheet contents on the clipboard. The Copy method has one
// optional parameter for specifying a destination. Because no argument
// is sent, the destination is the Clipboard.
workSheet.Range["A1:B3"].Copy();
4. Pressione CTRL+F5.
Um documento do Word é exibido contendo um ícone. Clique duas vezes no ícone para colocar a planilha
no primeiro plano.
((Excel.Range)workSheet.Columns[1]).AutoFit();
((Excel.Range)workSheet.Columns[2]).AutoFit();
2. Para alterar o padrão e usar PIAs em vez de inserir informações de tipo, expanda o nó Referências no
Gerenciador de Soluções e, em seguida, selecione Microsoft.Office.Interop.Excel ou
Microsoft.Office.Interop.Word.
3. Se você não conseguir ver a janela Propriedades, pressione F4.
4. Localize Inserir Tipos Interop na lista de propriedades e altere seu valor para False. De maneira
equivalente, você pode compilar usando a opção -Reference do compilador em vez de -link em um prompt
de comando.
O método AutoFormat tem sete parâmetros de valor, que são opcionais. Argumentos nomeados e
opcionais permitem que você forneça argumentos para nenhum, alguns ou todos eles. Na instrução
anterior, um argumento é fornecido para apenas um dos parâmetros, Format . Como Format é o primeiro
parâmetro na lista de parâmetros, você não precisará fornecer o nome do parâmetro. No entanto, poderá
ser mais fácil entender a instrução se o nome do parâmetro estiver incluído, conforme mostrado no código
a seguir.
2. Pressione CTRL + F5 para ver o resultado. Outros formatos estão listados na enumeração
XlRangeAutoFormat.
3. Compare a instrução na etapa 1 com o código a seguir, que mostra os argumentos que são necessários no
C# 3.0 ou versões anteriores.
// The AutoFormat method has seven optional value parameters. The
// following call specifies a value for the first parameter, and uses
// the default values for the other six.
Exemplo
O código a seguir mostra um exemplo completo.
using System;
using System.Collections.Generic;
using System.Linq;
using Excel = Microsoft.Office.Interop.Excel;
using Word = Microsoft.Office.Interop.Word;
namespace OfficeProgramminWalkthruComplete
{
class Walkthrough
{
static void Main(string[] args)
{
// Create a list of accounts.
var bankAccounts = new List<Account>
{
new Account {
ID = 345678,
Balance = 541.27
},
new Account {
ID = 1230221,
Balance = -127.44
}
};
var row = 1;
foreach (var acct in accounts)
{
row++;
workSheet.Cells[row, "A"] = acct.ID;
workSheet.Cells[row, "B"] = acct.Balance;
}
workSheet.Columns[1].AutoFit();
workSheet.Columns[2].AutoFit();
// Put the spreadsheet contents on the clipboard. The Copy method has one
// optional parameter for specifying a destination. Because no argument
// is sent, the destination is the Clipboard.
workSheet.Range["A1:B3"].Copy();
}
// The Add method has four reference parameters, all of which are
// optional. Visual C# allows you to omit arguments for them if
// the default values are what you want.
wordApp.Documents.Add();
Consulte também
Type.Missing
dynamic
Usando o tipo dynamic
Argumentos nomeados e opcionais
Como usar argumentos nomeados e opcionais na programação do Office
Como usar propriedades indexadas na programação
para interoperabilidade COM (Guia de Programação
em C#)
04/11/2019 • 3 minutes to read • Edit Online
As propriedades indexadas melhoram a maneira na qual as propriedades COM que têm parâmetros são
consumidas na programação em C#. As propriedades indexadas trabalham juntamente com outras
funcionalidades no Visual C#, como argumentos nomeados e opcionais, um novo tipo (dinâmico) e informações
de tipo inseridas para melhorar a programação do Microsoft Office.
Nas versões anteriores do C#, os métodos são acessíveis como propriedades apenas se o método get não tem
parâmetros e o método set tem apenas um parâmetro de valor. No entanto, nem todas as propriedades COM
atendem a essas restrições. Por exemplo, a propriedade Range[Object, Object] do Excel tem um acessador get
que requer um parâmetro para o nome do intervalo. No passado, como não era possível acessar a propriedade
Range diretamente, era necessário usar o método get_Range em vez disso, conforme mostrado no exemplo a
seguir.
// Visual C# 2010.
var excelApp = new Excel.Application();
// . . .
Excel.Range targetRange = excelApp.Range["A1"];
NOTE
O exemplo anterior também usa o recurso argumentos opcionais, que permite que você omita Type.Missing .
Da mesma forma, para definir o valor da propriedade Value de um objeto Range no C# 3.0 e versões anteriores,
são necessários dois argumentos. Um fornece um argumento para um parâmetro opcional que especifica o tipo
do valor de intervalo. O outro fornece o valor para a propriedade Value . Os exemplos a seguir ilustram essas
técnicas. Ambos definem o valor da célula A1 como Name .
// Visual C# 2008.
targetRange.set_Value(Type.Missing, "Name");
// Or
targetRange.Value2 = "Name";
// Visual C# 2010.
targetRange.Value = "Name";
Não é possível criar propriedades indexadas de sua preferência. O recurso dá suporte apenas ao consumo de
propriedades indexadas existentes.
Exemplo
O código a seguir mostra um exemplo completo. Para obter mais informações sobre como configurar um projeto
que acessa a API do Office, consulte Como acessar objetos de interoperabilidade do Office usando recursos do
Visual C#.
namespace IndexedProperties
{
class Program
{
static void Main(string[] args)
{
CSharp2010();
//CSharp2008();
}
}
}
}
Consulte também
Argumentos nomeados e opcionais
dynamic
Usando o tipo dynamic
Como usar argumentos nomeados e opcionais na programação do Office
Como acessar objetos de interoperabilidade do Office usando recursos do Visual C#
Passo a passo: programação do Office
Como usar invocação de plataforma para executar
um arquivo wave (Guia de Programação em C#)
30/10/2019 • 3 minutes to read • Edit Online
O exemplo de código C# a seguir ilustra como usar os serviços de invocação de plataforma para reproduzir um
arquivo de som wave no sistema operacional Windows.
Exemplo
Esse código de exemplo usa DllImportAttribute para importar o ponto de entrada de método PlaySound da
winmm.dll como Form1 PlaySound() . O exemplo tem um Windows Form simples com um botão. Ao clicar no
botão, abre-se uma caixa de diálogo padrão OpenFileDialog do Windows para que você possa abrir o arquivo
para reprodução. Quando um arquivo Wave é selecionado, ele é reproduzido usando o método PlaySound() da
biblioteca winmm. dll . Para obter mais informações sobre esse método, consulte Usando a função PlaySound
com arquivos de áudio Waveform. Procure e selecione um arquivo que tenha uma extensão .wav e, em seguida,
clique em Abrir para reproduzir o arquivo wave usando a invocação de plataforma. Uma caixa de texto exibe o
caminho completo do arquivo selecionado.
A caixa de diálogo Abrir Arquivos é filtrada por meio das configurações de filtro para mostrar somente os
arquivos que têm uma extensão .wav:
namespace WinSound
{
public partial class Form1 : Form
{
private TextBox textBox1;
private Button button1;
[System.Flags]
public enum PlaySoundFlags : int
{
SND_SYNC = 0x0000,
SND_ASYNC = 0x0001,
SND_NODEFAULT = 0x0002,
SND_LOOP = 0x0008,
SND_NOSTOP = 0x0010,
SND_NOWAIT = 0x00002000,
SND_FILENAME = 0x00020000,
SND_RESOURCE = 0x00040004
}
if (dialog1.ShowDialog() == DialogResult.OK)
{
textBox1.Text = dialog1.FileName;
PlaySound(dialog1.FileName, new System.IntPtr(), PlaySoundFlags.SND_SYNC);
}
}
Compilando o código
1. Crie um novo C# projeto de aplicativo Windows Forms no Visual Studio e nomeie-o WinSound.
2. Copie o código acima e cole-o sobre o conteúdo do arquivo Form1.cs .
3. Copie o código a seguir e cole-o no arquivo Form1.designer.cs , no método InitializeComponent() , após
qualquer código existente.
Consulte também
Guia de Programação em C#
Visão geral sobre interoperabilidade
Um olhar detalhado sobre invocação de plataforma
Marshaling de dados com a invocação de plataforma
Passo a passo: Programação do Office (C# e Visual
Basic)
04/11/2019 • 21 minutes to read • Edit Online
O Visual Studio oferece funcionalidades no C# e no Visual Basic que melhoram a programação do Microsoft
Office. As funcionalidades úteis do C# incluem argumentos nomeados e opcionais e valores retornados do tipo
dynamic . Na programação COM, você pode omitir a palavra-chave ref e obter acesso a propriedades
indexadas. As funcionalidades do Visual Basic incluem propriedades autoimplementadas, instruções em
expressões lambda e inicializadores de coleção.
Ambas as linguagens permitem incorporar as informações de tipo, que permitem a implantação de assemblies
que interagem com componentes COM sem implantar assemblies de interoperabilidade primários (PIAs) no
computador do usuário. Para obter mais informações, consulte Instruções passo a passo: Inserindo tipos de
assemblies gerenciados.
Este passo a passo demonstra essas funcionalidades no contexto de programação do Office, mas muitos deles
também são úteis na programação em geral. No passo a passo, você usa um aplicativo Suplemento do Excel para
criar uma pasta de trabalho do Excel. Em seguida, você cria um documento do Word que contém um link para a
pasta de trabalho. Por fim, você vê como habilitar e desabilitar a dependência de PIA.
Prerequisites
Você deve ter o Microsoft Office Excel e o Microsoft Office Word instalados no computador para concluir esse
passo a passo.
NOTE
Seu computador pode mostrar diferentes nomes ou locais para alguns dos elementos de interface do usuário do Visual
Studio nas instruções a seguir. A edição do Visual Studio que você possui e as configurações que você usa determinam
esses elementos. Para obter mais informações, consulte Personalizando o IDE.
using System.Collections.Generic;
using Excel = Microsoft.Office.Interop.Excel;
using Word = Microsoft.Office.Interop.Word;
Imports Microsoft.Office.Interop
class Account
{
public int ID { get; set; }
public double Balance { get; set; }
}
3. Para criar uma lista que contém duas contas, adicione o seguinte código ao método
bankAccounts
ThisAddIn_Startup em ThisAddIn.vb ou ThisAddIn.cs. As declarações de lista usam inicializadores de
coleção. Para obter mais informações, consulte Inicializadores de coleção.
var bankAccounts = new List<Account>
{
new Account
{
ID = 345,
Balance = 541.27
},
new Account
{
ID = 123,
Balance = -127.44
}
};
With Me.Application
' Add a new Excel workbook.
.Workbooks.Add()
.Visible = True
.Range("A1").Value = "ID"
.Range("B1").Value = "Balance"
.Range("A2").Select()
Dois novos recursos do C# são usados neste método. Esses dois recursos já existem no Visual Basic.
O método Add tem um parâmetro opcional para especificar um modelo específico. Os parâmetros
opcionais, novos no C# 4, permitem omitir o argumento para esse parâmetro se você quiser usar o
valor padrão do parâmetro. Como nenhum argumento é enviado no código anterior, Add usa o
modelo padrão e cria uma nova pasta de trabalho. A instrução equivalente em versões anteriores do
C# requer um argumento de espaço reservado: excelApp.Workbooks.Add(Type.Missing) .
Para obter mais informações, consulte Argumentos nomeados e opcionais.
As propriedades Range e Offset do objeto Range usam o recurso de propriedades indexadas. Este
recurso permite consumir essas propriedades de tipos COM usando a sintaxe típica do C# a seguir.
Propriedades indexadas também permitem que você use a propriedade Value do objeto Range ,
eliminando a necessidade de usar a propriedade Value2 . A propriedade Value é indexada, mas o
índice é opcional. Argumentos opcionais e propriedades indexadas trabalham juntos no exemplo a
seguir.
// In Visual C# 2008, you cannot access the Range, Offset, and Value
// properties directly.
excelApp.get_Range("A1").Value2 = "ID";
excelApp.ActiveCell.get_Offset(1, 0).Select();
Não é possível criar propriedades indexadas de sua preferência. O recurso dá suporte apenas ao
consumo de propriedades indexadas existentes.
Para obter mais informações, consulte Como usar propriedades indexadas na programação para
interoperabilidade COM.
2. Adicione o seguinte código no final de DisplayInExcel para ajustar as larguras das colunas para adequar o
conteúdo.
excelApp.Columns[1].AutoFit();
excelApp.Columns[2].AutoFit();
' Add the following two lines at the end of the With statement.
.Columns(1).AutoFit()
.Columns(2).AutoFit()
Essas adições demonstram outro recurso no C#: tratar valores Object retornados de hosts COM como o
Office, como se eles tivessem o tipo dinâmico. Isso ocorre automaticamente quando os tipos de
interoperabilidade de inserção são definidos com o valor padrão, True ou, de maneira equivalente,
quando o assembly é referenciado pela opção de compilador -link . O tipo dynamic permite a vinculação
posterior, já disponível no Visual Basic, e evita a conversão explícita necessária no C# 3.0 e em versões
anteriores da linguagem.
Por exemplo, excelApp.Columns[1] retorna um Object e AutoFit é um método Range do Excel. Sem
dynamic , você deve converter o objeto retornado em excelApp.Columns[1] como uma instância de Range
antes de chamar o método AutoFit .
Para obter mais informações sobre como inserir tipos de interoperabilidade, consulte os procedimentos
"Para localizar a referência de PIA" e "Para restaurar a dependência de PIA" posteriormente neste tópico.
Para obter mais informações sobre dynamic , consulte dynamic ou Usando o tipo dynamic.
Para invocar DisplayInExcel
1. Adicione o código a seguir no final do método ThisAddIn_StartUp . A chamada para DisplayInExcel
contém dois argumentos. O primeiro argumento é o nome da lista de contas a ser processada. O segundo
argumento é uma expressão lambda com várias linhas que define como os dados deverão ser
processados. Os valores ID e balance de cada conta serão exibidos em células adjacentes e a linha será
exibida em vermelho se o equilíbrio for menor do que zero. Para obter mais informações, consulte
Expressões Lambda.
2. Para executar o programa, pressione F5. Uma planilha do Excel é exibida contendo os dados das contas.
Para adicionar um documento do Word
1. Adicione o código a seguir ao final do método ThisAddIn_StartUp para criar um documento do Word que
contém um link para a pasta de trabalho do Excel.
Esse código demonstra vários dos novos recursos no C#: a capacidade para omitir a palavra-chave ref
em programação COM, argumentos nomeados e argumentos opcionais. Esses recursos já existem no
Visual Basic. O método PasteSpecial tem sete parâmetros, que são definidos como parâmetros de
referência opcionais. Argumentos opcionais e nomeados permitem designar os parâmetros que você
deseja acessar pelo nome e enviar argumentos apenas para esses parâmetros. Neste exemplo, os
argumentos são enviados para indicar que deve ser criado um link para a pasta de trabalho na área de
transferência (parâmetro Link ), e que o link será exibido no documento do Word como um ícone
(parâmetro DisplayAsIcon ). O Visual C# permite também que você omita a palavra-chave ref nesses
argumentos.
Para executar o aplicativo
1. Pressione F5 para executar o aplicativo. O Excel é iniciado e exibe uma tabela que contém as informações das
duas contas em bankAccounts . Em seguida, um documento do Word é exibido contendo um link para a tabela
do Excel.
Para limpar o projeto concluído
1. No Visual Studio, clique em Limpar Solução no menu Compilar. Caso contrário, o suplemento será
executado toda vez que você abrir o Excel no seu computador.
Para localizar a referência de PIA
1. Execute o aplicativo novamente, mas não clique em Limpar Solução.
2. Selecione Iniciar. Localize Microsoft Visual Studio <versão> e abra o prompt de comando do
desenvolvedor.
3. Digite ildasm na janela Prompt de Comando do Desenvolvedor para Visual Studio e pressione Enter. A
janela IL DASM é exibida.
4. No menu Arquivo na janela IL DASM, selecione Arquivo > Abrir. Clique duas vezes em Visual Studio
<versão> e clique duas vezes em Projetos. Abra a pasta do seu projeto e procure na pasta bin/Debug por
nome do projeto.dll. Clique duas vezes em nome do projeto.dll. Uma nova janela exibe os atributos do
projeto, além das referências a outros módulos e assemblies. Observe que os namespaces
Microsoft.Office.Interop.Excel e Microsoft.Office.Interop.Word estão incluídos no assembly. Por padrão,
no Visual Studio, o compilador importa os tipos que você precisa de um PIA referenciado para o seu
assembly.
Para obter mais informações, consulte Como exibir o conteúdo de um assembly.
5. Clique duas vezes no ícone MANIFEST. Uma janela será exibida contendo uma lista de assemblies que
contêm itens referenciados pelo projeto. Microsoft.Office.Interop.Excel e Microsoft.Office.Interop.Word
não estão incluídos na lista. Como os tipos do seu projeto precisam ter sido importados para o assembly,
referências a um PIA não são necessárias. Isso facilita a implantação. Os PIAs não precisam estar
presentes no computador do usuário e como um aplicativo não requer a implantação de uma versão
específica de um PIA, os aplicativos podem ser projetados para trabalhar com várias versões do Office,
desde que as APIs necessárias existam em todas as versões.
Como a implantação de PIAs não é mais necessária, você pode criar um aplicativo em cenários avançados
que funcione com várias versões do Office, incluindo versões anteriores. No entanto, isso funcionará
apenas se seu código não usar quaisquer APIs que não estejam disponíveis na versão do Office na qual
você está trabalhando. Não é sempre claro se uma determinada API estava disponível em uma versão
anterior e, por essa razão, não é recomendado trabalhar com versões anteriores do Office.
NOTE
O Office não publicou PIAs antes do Office 2003. Portanto, a única maneira de gerar um assembly de
interoperabilidade para o Office 2002 ou versões anteriores é importando referência COM.
A seguir está um exemplo de uma classe que você poderia expor como um objeto COM. Depois que esse código é
colocado em um arquivo .cs e adicionado ao seu projeto, defina a propriedade Registrar para
interoperabilidade COM como True. Para obter mais informações, confira Como: registrar um componente
para interoperabilidade COM.
A exposição de objetos do Visual C# para COM requer a declaração de uma interface de classe, de uma interface
de eventos se necessário e da própria classe. Os membros de classe devem seguir estas regras para ficarem
visíveis ao COM:
A classe deve ser pública.
As propriedades, os métodos e os eventos devem ser públicos.
As propriedades e os métodos devem ser declarados na interface de classe.
Os eventos devem ser declarados na interface de eventos.
Os outros membros públicos na classe que não forem declarados nessas interfaces não estarão visíveis para o
COM, mas eles estarão visíveis para outros objetos do .NET Framework.
Para expor propriedades e métodos ao COM, você deve declará-los na interface de classe e marcá-los com um
atributo DispId e implementá-los na classe. A ordem na qual os membros são declarados na interface é a ordem
usada para a vtable do COM.
Para expor eventos de sua classe, você deve declará-los na interface de eventos e marcá-los com um atributo
DispId . A classe não deve implementar essa interface.
A classe implementa a interface de classe. Ela pode implementar mais de uma interface, mas a primeira
implementação será a interface de classe padrão. Implemente os métodos e propriedades expostos ao COM aqui.
Eles devem ser marcados como públicos e devem corresponder às declarações na interface de classe. Além disso,
declare aqui os eventos acionados pela classe. Eles devem ser marcados como públicos e devem corresponder às
declarações na interface de eventos.
Exemplo
using System.Runtime.InteropServices;
namespace project_name
{
[Guid("EAA4976A-45C3-4BC5-BC0B-E474F4C3C83F")]
public interface ComClass1Interface
{
}
[Guid("7BD20046-DF8C-44A6-8F6B-687FAA26FA71"),
InterfaceType(ComInterfaceType.InterfaceIsIDispatch)]
public interface ComClass1Events
{
}
[Guid("0D53A3E8-E51A-49C7-944E-E72A2064F938"),
ClassInterface(ClassInterfaceType.None),
ComSourceInterfaces(typeof(ComClass1Events))]
public class ComClass1 : ComClass1Interface
{
}
}
Consulte também
Guia de Programação em C#
Interoperabilidade
Página de Build, Designer de Projeto (C#)
C#referência
27/11/2019 • 6 minutes to read • Edit Online
Nesta seção
Palavras-chave do C#
Fornece links para informações sobre a sintaxe e as palavras-chave do C#.
Operadores do C#
Fornece links para informações sobre a sintaxe e os operadores do C#.
Caracteres especiais de C#
Fornece links para informações sobre caracteres especiais contextuais em C# e seu uso.
Diretivas do pré-processador do C#
Fornece links para informações sobre os comandos do compilador para inserir no código-
fonte do C#.
Opções do compilador de C#
Inclui informações sobre as opções do compilador e como usá-las.
Erros do Compilador do C#
Inclui snippets de código que demonstram a causa e a correção de erros do compilador do
C# e avisos.
C# Language Specification (Especificação da linguagem C#)
Especificação de linguagem do C# 6.0. Este é um projeto de proposta da linguagem C# 6.0.
Este documento será refinado por meio do trabalho C# com o Comitê de padrões ECMA. A
versão 5.0 foi lançada em dezembro de 2017 como o documento Padrão ECMA-334 – 5ª
Edição.
Os recursos que foram implementados nas versões do C# depois da 6.0 são representados
em propostas de especificação de linguagem. Esses documentos descrevem os deltas para a
especificação da linguagem a fim de adicionar os novos recursos. Eles estão no formulário de
proposta de rascunho. Essas especificações serão refinadas e enviadas para o Comitê de
padrões ECMA para revisão formal e Incorporation em uma versão futura C# do padrão.
C#7,0 propostas de especificação
Implementamos diversos recursos novos no C# 7.0. Entre eles estão correspondência de
padrões, funções locais, declarações de variável out, expressões throw, literais binários e
separadores de dígito. Esta pasta contém as especificações de cada um desses recursos.
C#7,1 propostas de especificação
Há novos recursos adicionados no C# 7.1. Primeiramente, você pode gravar um método
Main que retorna Task ou Task<int> . Isso permite que você adicione o modificador async
ao Main . A expressão default pode ser usada sem um tipo em locais onde o tipo pode ser
inferido. Além disso, os nomes dos membros de tupla podem ser inferidos. Por fim, a
correspondência de padrões pode ser usada com genéricos.
C#7,2 propostas de especificação
O C# 7.2 adicionou a uma série de recursos pequenos. Você pode passar argumentos por
referência de somente leitura usando a palavra-chave in . Há uma série de alterações de
nível inferior para dar suporte à segurança de tempo de compilação para Span e tipos
relacionados. Você pode usar argumentos nomeados nos quais os argumentos posteriores
são posicionais, em algumas situações. O modificador de acesso private protected permite
que você especifique que os chamadores são limitados aos tipos derivados, implementados
no mesmo assembly. O operador ?: pode resolver em uma referência a uma variável. Você
também pode formatar números hexadecimais e binários usando um separador de dígito à
esquerda.
C#7,3 propostas de especificação
C# 7.3 é outra versão de ponto que inclui várias atualizações pequenas. Você pode usar
novas restrições em parâmetros de tipo genérico. Outras alterações facilitam trabalhar com
campos fixed , incluindo o uso de alocações stackalloc . Variáveis locais declaradas com a
palavra-chave ref podem ser reatribuídas para se referirem ao novo armazenamento. Você
pode colocar os atributos em propriedades autoimplementadas que direcionam o campo de
suporte gerado pelo compilador. As variáveis de expressão podem ser usadas em
inicializadores. As tuplas podem ser comparadas quanto à igualdade (ou desigualdade).
Também houve algumas melhorias para a resolução de sobrecarga.
C#8,0 propostas de especificação
C#8,0 está disponível com o .NET Core 3,0. Os recursos incluem tipos de referência
anuláveis, correspondência de padrões recursivos, métodos de interface padrão, fluxos
assíncronos, intervalos e índices, com base em padrões usando e usando declarações,
atribuição de União nula e membros de instância ReadOnly.
Seções Relacionadas
Usando o Ambiente de Desenvolvimento do Visual Studio para C#
Fornece links para tópicos conceituais e de tarefas que descrevem o IDE e o Editor.
Guia de Programação em C#
Inclui informações sobre como usar a linguagem de programação do C#.
Controle de versão da linguagem C#
23/10/2019 • 6 minutes to read • Edit Online
O compilador do C# mais recente determina uma versão da linguagem padrão com base nas estruturas de
destino do projeto. Isso ocorre porque a linguagem C# pode ter recursos que dependem de tipos ou de
componentes de tempo de execução que não estão disponíveis em todas as implementações do .NET. Com
isso, independentemente do destino no qual seu projeto é criado, você obtém a versão da linguagem mais
compatível por padrão.
As regras deste artigo se aplicam ao compilador fornecido com o Visual Studio 2019 ou o SDK do .NET Core
3.0. Os compiladores do C# que fazem parte da instalação do Visual Studio 2017 ou de versões anteriores do
SDK do .NET Core são direcionados ao C# 7.0 por padrão.
Padrão
O compilador determina um padrão com base nestas regras:
Substituir um padrão
Se precisar especificar sua versão do C# explicitamente, poderá fazer isso de várias maneiras:
Edite manualmente o arquivo de projeto.
Definir a versão da linguagem para vários projetos em um subdiretório.
Configurar a opção -langversion do compilador
Editar o arquivo de projeto
É possível definir a versão da linguagem em seu arquivo de projeto. Por exemplo, se você quiser
explicitamente acesso às versões prévias dos recursos, adicione um elemento como este:
<PropertyGroup>
<LangVersion>preview</LangVersion>
</PropertyGroup>
O valor preview usa a versão prévia mais recente da linguagem C# compatível com seu compilador.
Configurar vários projetos
Crie um arquivo Directory.Build.props que contém o elemento <LangVersion> para configurar vários
diretórios. Normalmente, você faz isso no diretório da solução. Adicione o seguinte a um arquivo
Directory.Build.props no diretório de solução:
<Project>
<PropertyGroup>
<LangVersion>preview</LangVersion>
</PropertyGroup>
</Project>
Agora, os builds de todo subdiretório do diretório que contém esse arquivo usarão a versão do C# da versão
prévia. Para obter mais informações, confira o artigo sobre como personalizar o build.
VALOR SIGNIFICADO
volatile while
Palavras-chave contextuais
Uma palavra-chave contextual é usada para fornecer um significado específico no código, mas não é
uma palavra reservada no C#. Algumas palavras-chave contextuais, como partial e where , têm
significados especiais em dois ou mais contextos.
async await by
let nameof on
when (condição de filtro) where (restrição de tipo genérico) where (cláusula de consulta)
yield
Consulte também
Referência de C#
Tipos de valor (Referência de C#)
29/11/2019 • 5 minutes to read • Edit Online
Tipos simples
Os tipos simples são um conjunto de tipos de struct predefinidos fornecidos por C# e incluem os seguintes
tipos:
Tipos integrais: tipos numéricos inteiros e o tipo char
Tipos de ponto flutuante
bool
Os tipos simples são identificados por meio de palavras-chave, mas essas palavras-chave são simplesmente
aliases para tipos de struct predefinidos no namespace System. Por exemplo, int é um alias de System.Int32.
Para obter uma lista completa de aliases, consulte Tabela de tipos internos.
Os tipos simples diferem de outros tipos de struct, pois permitem determinadas operações adicionais:
Os tipos simples podem ser inicializados com o uso de literais. Por exemplo, 'A' é um literal do tipo
char e 2001 é um literal do tipo int .
Você pode declarar constantes dos tipos simples com a palavra-chave const. Não é possível ter
constantes de outros tipos de struct.
As expressões constantes, cujos operandos são todos constantes de tipo simples, são avaliadas em
tempo de compilação.
Para saber mais, confira a seção Tipos simples na Especificação da linguagem C#.
Inicializando tipos de valor
As variáveis locais no C# devem ser inicializadas antes de serem usadas. Por exemplo, você pode declarar
uma variável local sem inicialização, como no exemplo a seguir:
int myInt;
Você não pode usá-la antes de inicializá-la. Você pode inicializar a variável usando a instrução a seguir:
É claro que a declaração e a inicialização podem ser feitas na mesma instrução, como nos exemplos a
seguir:
– ou –
int myInt = 0;
Usando o operador new, chama o construtor sem parâmetro do tipo específico e atribui o valor padrão à
variável. No exemplo anterior, o construtor sem parâmetro atribuiu o valor 0 para myInt . Para saber mais
sobre valores atribuídos ao chamar construtores sem parâmetros, confira Tabela de valores padrão.
Com tipos definidos pelo usuário, use new para invocar o construtor sem parâmetro. Por exemplo, a
instrução a seguir invoca o construtor sem parâmetro do struct Point :
Após esta chamada, o struct é considerado para ser definitivamente atribuído, ou seja, todos os seus
membros são inicializados com seus valores padrão.
Para saber mais sobre o operador new , confira new.
Para saber mais sobre a formatação da saída de tipos numéricos, consulte Tabela de formatação de
resultados numéricos.
Consulte também
Referência de C#
Palavras-chave do C#
Tipos de referência
Tipos de valor anuláveis
Tipos numéricos integrais (Referência
C#)
30/10/2019 • 6 minutes to read • Edit Online
PALAVRA-
CHAVE/TIPO C# INTERVALO TAMANHO TIPO .NET
int a = 123;
System.Int32 b = 123;
O valor padrão de cada tipo integral é zero, 0 . Cada um dos tipos integrais possui
as constantes MinValue e MaxValue que fornecem o valor mínimo e máximo desse
tipo.
Use a estrutura System.Numerics.BigInteger para representar um inteiro com sinal
sem nenhum limite superior ou inferior.
Literais inteiros
Literais inteiros podem ser
decimal: sem nenhum prefixo
hexadecimal: com o prefixo de 0x ou 0X
Binary: com o prefixo 0b ou 0B (disponível em C# 7,0 e posterior)
O código a seguir demonstra um exemplo de cada um:
NOTE
Você pode usar a letra minúscula l como um sufixo. No entanto, isso gera um
aviso do compilador porque a letra l pode ser confundida com o 1 de dígitos.
Use L para maior clareza.
byte a = 17;
byte b = 300; // CS0031: Constant value '300' cannot be converted to a 'byte'
Como mostra o exemplo anterior, se o valor do literal não estiver dentro do
intervalo do tipo de destino, ocorrerá um erro de compilador CS0031 .
Você também pode usar uma conversão para converter o valor representado por
um literal inteiro para o tipo diferente do tipo determinado do literal:
Conversões
Você pode converter qualquer tipo numérico integral para qualquer outro tipo
numérico integral. Se o tipo de destino puder armazenar todos os valores do tipo
de origem, a conversão será implícita. Caso contrário, você precisa usar o operador
cast () para invocar uma conversão explícita. Para obter mais informações,
consulte conversões numéricas internas.
Especificação da linguagem C#
Para obter mais informações, confira as seguintes seções da especificação da
linguagem C#:
Tipos integrais
Literais inteiros
Consulte também
Referência de C#
Tabela de tipos internos
Tipos de ponto flutuante
Tabela de formatação de resultados numéricos
Numéricos no .NET
Tipos numéricos de ponto flutuante (Referência
de C#)
29/11/2019 • 6 minutes to read • Edit Online
Os tipos de ponto flutuante são um subconjunto dos tipos simples e podem ser inicializados com
literais. Todos os tipos de ponto flutuante também são tipos de valor. Todos os tipos numéricos de ponto
flutuante dão suporte a operadores aritméticos, de comparaçãoe de igualdade .
PALAVRA- INTERVALO
CHAVE/TIPO C# APROXIMADO PRECISION TAMANHO TIPO .NET
Na tabela anterior, cada palavra-chave do tipo C# da coluna mais à esquerda é um alias do tipo .NET
correspondente. Eles são intercambiáveis. Por exemplo, as declarações a seguir declaram variáveis do
mesmo tipo:
double a = 12.3;
System.Double b = 12.3;
O valor padrão de cada tipo de ponto flutuante é zero, 0 . Cada um dos tipos de ponto flutuante possui
as constantes MinValue e MaxValue que fornecem o valor mínimo e máximo desse tipo. Os tipos float
e double também fornecem constantes que representam valores não numéricos e infinitos. Por exemplo,
o tipo double fornece as seguintes constantes: Double.NaN, Double.NegativeInfinity e
Double.PositiveInfinity.
Como o tipo decimal tem mais precisão e um intervalo menor que float e double , ele é apropriado
para cálculos financeiros e monetários.
É possível misturar tipos integrais e de ponto flutuante em uma expressão. Nesse caso, os tipos integrais
são convertidos em tipos de ponto flutuante. A avaliação da expressão é executada de acordo com as
regras a seguir:
Se um dos tipos de ponto flutuante for double , a expressão será avaliada como double ou bool em
comparações relacionais e de igualdade.
Se não houver nenhum tipo de double na expressão, a expressão será avaliada como float ou bool
em comparações relacionais e de igualdade.
Uma expressão de ponto flutuante pode conter os seguintes conjuntos de valores:
Zero positivo e negativo
Infinito positivo e negativo
Valor NaN (não é um número)
O conjunto finito de valores diferentes de zero
Para obter mais informações sobre esses valores, consulte o padrão IEEE para Aritmética de ponto
flutuante binário, disponível no site do IEEE.
É possível usar cadeias de caracteres de formato numérico padrão ou cadeias de caracteres de formato
numérico personalizado para formatar um valor de ponto flutuante.
Literais reais
O tipo de um literal real é determinado pelo seu sufixo da seguinte maneira:
O literal sem sufixo ou com o sufixo d ou D é do tipo double
O literal com o sufixo f ou F é do tipo float
O literal com o sufixo m ou M é do tipo decimal
O código a seguir demonstra um exemplo de cada um:
double d = 3D;
d = 4d;
d = 3.934_001;
float f = 3_000.5F;
f = 5.4f;
O exemplo anterior também mostra o uso de _ como um separador de dígito, que tem C# suporte a
partir de 7,0. Você pode usar o separador de dígitos com todos os tipos de literais numéricos.
Você também pode usar a notação científica, ou seja, especificar uma parte exponencial de um literal real,
como mostra o exemplo a seguir:
double d = 0.42e2;
Console.WriteLine(d); // output 42;
float f = 134.45E-2f;
Console.WriteLine(f); // output: 1.3445
decimal m = 1.5E6m;
Console.WriteLine(m); // output: 1500000
Conversões
Há apenas uma conversão implícita entre tipos numéricos de ponto flutuante: de float para double . No
entanto, você pode converter qualquer tipo de ponto flutuante para qualquer outro tipo de ponto
flutuante com a conversão explícita. Para obter mais informações, consulte conversões numéricas
internas.
Especificação da linguagem C#
Para obter mais informações, confira as seguintes seções da especificação da linguagem C#:
Tipos de ponto flutuante
O tipo decimal
Literais reais
Consulte também
Referência de C#
Tabela de tipos internos
Tipos integrais
Tabela de formatação de resultados numéricos
Cadeias de caracteres de formato numérico padrão
Numéricos no .NET
System.Numerics.Complex
Conversões numéricas internas (C# referência)
24/10/2019 • 8 minutes to read • Edit Online
C#fornece um conjunto de tipos numéricos de ponto flutuante e integral . Existe uma conversão entre dois tipos
numéricos, implícito ou explícito. Você deve usar o operador cast () para invocar uma conversão explícita.
DE PARA
float double
NOTE
As conversões implícitas de int , uint , long ou ulong para float e de long ou ulong para double podem
causar uma perda de precisão, mas nunca uma perda de uma ordem de magnitude. As outras conversões numéricas
implícitas nunca perdem nenhuma informação.
byte a = 13;
byte b = 300; // CS0031: Constant value '300' cannot be converted to a 'byte'
Como mostra o exemplo anterior, se o valor da constante não estiver dentro do intervalo do tipo de
destino, ocorrerá um erro de compilador CS0031 .
DE PARA
byte sbyte
NOTE
Uma conversão numérica explícita pode resultar em perda de dados ou gerar uma exceção, normalmente uma
OverflowException.
Especificação da linguagem C#
Para obter mais informações, confira as seguintes seções da especificação da linguagem C#:
Conversões numéricas implícitas
Conversões numéricas explícitas
Consulte também
Referência de C#
Conversão e conversões de tipo
2 minutes to read
Char (C# referência)
29/11/2019 • 2 minutes to read • Edit Online
A palavra-chave Type de char é um alias para o tipo de estrutura .NET System.Char que representa um
caractere Unicode UTF -16.
{1>Literais<1}
Você pode especificar um valor de char com:
um literal de caractere.
uma sequência de escape Unicode, que é \u seguida pela representação hexadecimal de quatro símbolos
de um código de caractere.
uma sequência de escape hexadecimal, que é \x seguida pela representação hexadecimal de um código de
caractere.
Como mostra o exemplo anterior, você também pode converter o valor de um código de caractere no valor de
char correspondente.
NOTE
No caso de uma sequência de escape Unicode, você deve especificar todos os quatro dígitos hexadecimais. Ou seja,
\u006A é uma sequência de escape válida, enquanto \u06A e \u6A não são válidos.
No caso de uma sequência de escape hexadecimal, você pode omitir os zeros à esquerda. Ou seja, as sequências de
escape \x006A , \x06A e \x6A são válidas e correspondem ao mesmo caractere.
Conversões
O tipo de char é implicitamente conversível para os seguintes tipos integral : ushort , int , uint , long e
ulong . Ele também é implicitamente conversível para os tipos numéricos de ponto flutuante internos: float ,
double e decimal . Ele é explicitamente conversível para sbyte , byte e short tipos integrais.
Não há conversões implícitas de outros tipos para o tipo de char . No entanto, qualquer tipo integral ou
numérico de ponto flutuante é explicitamente conversível para char .
Especificação da linguagem C#
Para obter mais informações, consulte a seção tipos integrais da C# especificação da linguagem.
Consulte também
Referência de C#
Tabela de tipos internos
Cadeias de Caracteres
enum (Referência de C#)
27/11/2019 • 7 minutes to read • Edit Online
A palavra-chave enum é usada para declarar uma enumeração, um tipo distinto que consiste em um
conjunto de constantes nomeadas denominado lista de enumeradores.
Normalmente, é melhor definir um enum diretamente dentro de um namespace para que todas as
classes no namespace possam acessá-lo com a mesma conveniência. No entanto, um enum também
pode ser aninhado dentro de uma classe ou struct.
Por padrão, o primeiro enumerador tem o valor 0 e o valor de cada enumerador seguinte é aumentado
em 1. Por exemplo, na seguinte enumeração, Sat é 0 , Sun é 1 , Mon é 2 e assim por diante.
Enumeradores podem usar inicializadores para substituir os valores padrão, conforme mostrado no
exemplo a seguir.
enum Day : byte {Sat=1, Sun, Mon, Tue, Wed, Thu, Fri};
Uma variável de um tipo de enumeração pode receber qualquer valor no intervalo do tipo subjacente; os
valores não são limitados às constantes nomeadas.
O valor padrão de um enum E é o valor produzido pela expressão (E)0 .
NOTE
Um enumerador não pode conter espaços em branco em seu nome.
O tipo subjacente especifica quanto armazenamento é alocado para cada enumerador. No entanto, uma
conversão explícita é necessária para converter de um tipo enum em um tipo integral. Por exemplo, a
instrução a seguir atribui o enumerador Sun a uma variável do tipo int usando uma conversão para
converter de enum em int .
int x = (int)Day.Sun;
Quando você aplica System.FlagsAttribute a uma enumeração que contém elementos que podem ser
combinados com uma operação OR bit a bit, o atributo afeta o comportamento do enum quando ele é
usado com algumas ferramentas. Você pode observar essas alterações quando usa ferramentas como os
métodos da classe Console e o Avaliador de Expressão. (Consulte o terceiro exemplo.)
Programação robusta
Assim como ocorre com qualquer constante, todas as referências aos valores individuais de um enum
são convertidas em literais numéricos em tempo de compilação. Isso pode criar problemas de controle
de versão, conforme descrito em Constantes.
Atribuir valores adicionais a novas versões de enums ou alterar os valores dos membros de enum em
uma nova versão pode causar problemas para códigos-fonte dependentes. Valores de enum
frequentemente são usados em instruções switch. Se elementos adicionais tiverem sido adicionados ao
tipo enum , a seção padrão da instrução switch pode ser selecionada inesperadamente.
Se outros desenvolvedores usarem seu código, você deverá fornecer diretrizes sobre como seu código
deve reagir se novos elementos forem adicionados a qualquer tipo enum .
Exemplo
No exemplo a seguir, uma enumeração, Day , é declarada. Dois enumeradores são convertidos
explicitamente em inteiros e atribuídos a variáveis de inteiro.
Exemplo
No exemplo a seguir, a opção de tipo base é usada para declarar uma enum cujos membros são do tipo
long . Observe que, mesmo que o tipo subjacente da enumeração seja long , os membros da
enumeração ainda devem ser convertidos explicitamente no tipo long usando uma conversão.
public class EnumTest2
{
enum Range : long { Max = 2147483648L, Min = 255L };
static void Main()
{
long x = (long)Range.Max;
long y = (long)Range.Min;
Console.WriteLine("Max = {0}", x);
Console.WriteLine("Min = {0}", y);
}
}
/* Output:
Max = 2147483648
Min = 255
*/
Exemplo
O exemplo de código a seguir ilustra o uso e o efeito do atributo System.FlagsAttribute em uma
declaração enum .
class FlagTest
{
static void Main()
{
// The bitwise OR of 0001 and 0100 is 0101.
CarOptions options = CarOptions.SunRoof | CarOptions.FogLights;
Comments
Se você remover Flags , o exemplo exibirá os seguintes valores:
5
5
especificação da linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é
a fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Tipos de enumeração
Palavras-chave do C#
Tipos integrais
Tabela de tipos internos
Convenções de Nomenclatura do Enum
struct (Referência de C#)
04/11/2019 • 2 minutes to read • Edit Online
O tipo struct é um tipo de valor normalmente usado para encapsular pequenos grupos de
variáveis relacionadas, tais como coordenadas de um retângulo ou as características de um item
em um inventário. O seguinte exemplo mostra uma declaração simples de struct:
Comentários
Os structs também podem conter construtores, constantes, campos, métodos, propriedades,
indexadores, operadores, eventos e tipos aninhados, embora, se vários desses membros forem
necessários, você deva considerar tonar seu tipo uma classe.
Para obter exemplos, consulte Usando structs.
Os structs podem implantar uma interface, mas não podem herdá-la de outra struct. Por esse
motivo, os membros de struct não podem ser declarados como protected .
Para obter mais informações, consulte Structs.
Exemplos
Para obter exemplos e mais informações, consulte Usando structs.
Especificação da linguagem C#
Para obter exemplos, consulte Usando structs.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Tabela de valores padrão
Tabela de tipos internos
Tipos
Tipos de valor
class
interface
Classes e Structs
Tipos de valores anuláveis (C# referência)
08/11/2019 • 13 minutes to read • Edit Online
Um tipo de valor anulável T? representa todos os valores de seu tipo de valor subjacente T e um valor
nulo adicional. Por exemplo, você pode atribuir qualquer um dos três valores a seguir a uma bool?
variável: true , false ou null . Um tipo de valor subjacente T não pode ser um tipo de valor anulável em
si.
NOTE
C#8,0 apresenta o recurso de tipos de referência anulável. Para obter mais informações, consulte tipos de referência
anuláveis. Os tipos de valor anulável estão disponíveis a C# partir de 2.
Qualquer tipo de valor anulável é uma instância da estrutura de System.Nullable<T> genérica. Você pode
se referir a um tipo de valor anulável com um tipo subjacente T em qualquer um dos seguintes
formulários intercambiáveis: Nullable<T> ou T? .
Normalmente, você usa um tipo de valor anulável quando precisa representar o valor indefinido de um tipo
de valor subjacente. Por exemplo, um booliano ou bool variável só pode ser true ou false . No entanto,
em alguns aplicativos, um valor de variável pode ser indefinido ou estar ausente. Por exemplo, um campo
de banco de dados pode conter true ou false ou não pode conter nenhum valor, ou seja, NULL . Você
pode usar o tipo de bool? nesse cenário.
Declaração e atribuição
Como um tipo de valor é implicitamente conversível para o tipo de valor anulável correspondente, você
pode atribuir um valor a uma variável de um tipo de valor anulável, como faria com o tipo de valor
subjacente. Você também pode atribuir o valor null . Por exemplo:
double? pi = 3.14;
char? letter = 'a';
int m2 = 10;
int? m = m2;
O valor padrão de um tipo de valor anulável representa null , ou seja, é uma instância cuja propriedade
Nullable<T>.HasValue retorna false .
Você sempre pode usar as seguintes propriedades somente leitura para examinar e obter um valor de uma
variável de tipo de valor anulável:
Nullable<T>.HasValue indica se uma instância de um tipo de valor anulável tem um valor de seu tipo
subjacente.
Nullable<T>.Value obtém o valor de um tipo subjacente quando HasValue é true . Quando
HasValue é false , a propriedade Value gera uma InvalidOperationException.
O exemplo a seguir usa a propriedade HasValue para testar se a variável contém um valor antes de exibi-la:
int? b = 10;
if (b.HasValue)
{
Console.WriteLine($"b is {b.Value}");
}
else
{
Console.WriteLine("b does not have a value");
}
// Output:
// b is 10
Você também pode comparar uma variável de um tipo de valor anulável com null em vez de usar a
propriedade HasValue , como mostra o exemplo a seguir:
int? c = 7;
if (c != null)
{
Console.WriteLine($"c is {c.Value}");
}
else
{
Console.WriteLine("c does not have a value");
}
// Output:
// c is 7
int? c = null;
int d = c ?? -1;
Console.WriteLine($"d is {d}"); // output: d is -1
Se você quiser usar o valor padrão do tipo de valor subjacente no lugar de null , use o método
Nullable<T>.GetValueOrDefault().
Você também pode converter explicitamente um tipo de valor anulável para um tipo não anulável, como
mostra o exemplo a seguir:
int? n = null;
Em tempo de execução, se o valor de um tipo de valor anulável for null , a conversão explícita lançará um
InvalidOperationException.
Um tipo de valor não anulável T é implicitamente conversível para o tipo de valor anulável correspondente
T? .
Operadores levantados
Os operadores unários e binários predefinidos ou quaisquer operadores sobrecarregados com suporte de
um tipo de valor T também são suportados pelo tipo de valor anulável correspondente T? . Esses
operadores, também conhecidos como operadores levantados, produzem null se um ou ambos os
operandos forem null ; caso contrário, o operador usa os valores contidos de seus operandos para calcular
o resultado. Por exemplo:
int? a = 10;
int? b = null;
int? c = 10;
a++; // a is 11
a = a * c; // a is 110
a = a + b; // a is null
NOTE
Para o tipo de bool? , os operadores predefinidos & e | não seguem as regras descritas nesta seção: o resultado
de uma avaliação de operador pode ser não nulo, mesmo que um dos operandos seja null . Para obter mais
informações, confira a seção Operadores lógicos booleanos anuláveis do artigo Operadores lógicos boolianos.
Para os operadores de comparação < , > , <= e >= , se um ou ambos os operandos forem null , o
resultado será false ; caso contrário, os valores contidos dos operandos serão comparados. Não presuma
que como uma comparação (por exemplo, <= ) retorna false , a comparação oposta ( > ) retorna true . O
exemplo a seguir mostra que 10
Nem maior ou igual a null
Nem menor que null
int? a = 10;
Console.WriteLine($"{a} >= null is {a >= null}");
Console.WriteLine($"{a} < null is {a < null}");
Console.WriteLine($"{a} == null is {a == null}");
// Output:
// 10 >= null is False
// 10 < null is False
// 10 == null is False
int? b = null;
int? c = null;
Console.WriteLine($"null >= null is {b >= c}");
Console.WriteLine($"null == null is {b == c}");
// Output:
// null >= null is False
// null == null is True
O exemplo anterior também mostra que uma comparação de igualdade de duas instâncias de tipo de valor
anulável que são null são avaliadas como true .
Se existir uma conversão definida pelo usuário entre dois tipos de valor, a mesma conversão também
poderá ser usada entre os tipos de valores anuláveis correspondentes.
int a = 41;
object aBoxed = a;
int? aNullable = (int?)aBoxed;
Console.WriteLine($"Value of aNullable: {aNullable}");
// Output:
// int? is nullable type
// int is non-nullable type
Como mostra o exemplo, você usa o operador typeof para criar uma instância de System.Type.
Se você quiser determinar se uma instância é de um tipo de valor anulável, não use o método
Object.GetType para obter uma instância de Type a ser testada com o código anterior. Quando você chama
o método Object.GetType em uma instância de um tipo de valor anulável, a instância é encaixada para
Object. Como Boxing de uma instância não nula de um tipo de valor anulável é equivalente à Boxing de um
valor do tipo subjacente, GetType retorna uma instância de Type que representa o tipo subjacente de um
tipo de valor anulável:
int? a = 17;
Type typeOfA = a.GetType();
Console.WriteLine(typeOfA.FullName);
// Output:
// System.Int32
Além disso, não use o operador is para determinar se uma instância é de um tipo de valor anulável. Como
mostra o exemplo a seguir, você não pode distinguir tipos de uma instância de tipo de valor anulável e sua
instância de tipo subjacente com o operador de is :
int? a = 14;
if (a is int)
{
Console.WriteLine("int? instance is compatible with int");
}
int b = 17;
if (b is int?)
{
Console.WriteLine("int instance is compatible with int?");
}
// Output:
// int? instance is compatible with int
// int instance is compatible with int?
Você pode usar o código apresentado no exemplo a seguir para determinar se uma instância é de um tipo
de valor anulável:
int? a = 14;
Console.WriteLine(IsOfNullableType(a)); // output: True
int b = 17;
Console.WriteLine(IsOfNullableType(b)); // output: False
bool IsOfNullableType<T>(T o)
{
var type = typeof(T);
return Nullable.GetUnderlyingType(type) != null;
}
NOTE
Os métodos descritos nesta seção não são aplicáveis no caso de tipos de referência anuláveis.
Especificação da linguagem C#
Para obter mais informações, confira as seguintes seções da especificação da linguagem C#:
Tipos que permitem valor nulo
Operadores levantados
Conversões anuláveis implícitas
Conversões anuláveis explícitas
Operadores de conversão levantados
Consulte também
Referência de C#
O que exatamente "levantado" significa?
System.Nullable<T>
System.Nullable
Nullable.GetUnderlyingType
Tipos de referência nula
Tipos de referência (Referência em C#)
04/11/2019 • 2 minutes to read • Edit Online
Há dois tipos em C#: tipos de referência e valor. Variáveis de tipos de referência armazenam referências em
seus dados (objetos) enquanto que variáveis de tipos de valor contém diretamente seus dados. Com tipos de
referência, duas variáveis podem fazer referência ao mesmo objeto; portanto, operações em uma variável
podem afetar o objeto referenciado pela outra variável. Com tipos de valor, cada variável tem sua própria
cópia dos dados e as operações em uma variável não podem afetar a outra (exceto no caso das variáveis de
parâmetros in, ref e out. Confira o modificador de parâmetro in, ref e out).
As seguintes palavras-chaves são usadas para declarar tipos de referência:
class
interface
delegate
O C# também oferece os seguintes tipos de referência internos:
dynamic
object
string
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Tipos
Tipos de valor
Tipos de referência internos (Referência de
C#)
25/11/2019 • 12 minutes to read • Edit Online
O tipo de objeto
O tipo object é um alias de System.Object no .NET. No sistema de tipos unificado do C#,
todos os tipos, predefinidos e definidos pelo usuário, tipos de referência e tipos de valor,
herdam direta ou indiretamente de System.Object. Você pode atribuir valores de qualquer tipo
a variáveis do tipo object . Qualquer variável object pode ser atribuída ao seu valor padrão
usando o literal null . Quando uma variável de um tipo de valor é convertida para um objeto,
ela é chamada de boxed. Quando uma variável do objeto do tipo é convertida para um tipo de
valor, ela é chamada de unboxed. Para obter mais informações, consulte Boxing e unboxing.
string a = "hello";
string b = "h";
// Append to contents of 'b'
b += "ello";
Console.WriteLine(a == b);
Console.WriteLine(object.ReferenceEquals(a, b));
Isso exibe "True" e, em seguida, "False" porque os conteúdos das cadeias de caracteres são
equivalentes, mas a e b não fazem referência à mesma instância da cadeia de caracteres.
O operador + concatena as cadeias de caracteres:
O operador [] pode ser usado para acesso somente leitura a caracteres individuais de um
string . Os valores válidos começam em 0 e devem ser menores do que o comprimento do
string :
De maneira semelhante, o operador [] também pode ser usado para iterar em cada caractere
em um string :
Literais de cadeia de caracteres são do tipo string e podem ser escritos de duas formas, entre
aspas e @ . Os literais de cadeia de caracteres entre aspas são colocados entre aspas duplas ("):
NOTE
O código de escape \udddd (em que dddd é um número de quatro dígitos) representa o caractere
Unicode U+ dddd . Os códigos de escape Unicode de oito dígitos também são reconhecidos:
\Udddddddd .
Os literais de cadeia de caracteres textuais começam com @ e também são colocados entre
aspas duplas. Por exemplo:
A vantagem das cadeias de caracteres textuais é que as sequências de escape não são
processadas, o que torna mais fácil escrever, por exemplo, um nome de arquivo totalmente
qualificado do Windows:
@"c:\Docs\Source\a.txt" // rather than "c:\\Docs\\Source\\a.txt"
O tipo de delegado
A declaração de um tipo de delegado é semelhante a uma assinatura de método. Ela tem um
valor retornado e parâmetros de qualquer tipo:
O tipo dinâmico
O tipo dynamic indica que o uso de variável e as referências aos seus membros ignoram a
verificação de tipo de tempo de compilação. Em vez disso, essas operações são resolvidas em
tempo de execução. O tipo dynamic simplifica o acesso a APIs COM, como as APIs de
Automação do Office e também às APIs dinâmicas, como bibliotecas do IronPython e ao DOM
(Modelo de Objeto do Documento) HTML.
O tipo dynamic se comporta como o tipo object na maioria das circunstâncias. Em particular,
qualquer expressão não nula pode ser convertida no tipo dynamic . O tipo dynamic é diferente
de object nas operações que contêm expressões do tipo dynamic não resolvidas ou com tipo
verificado pelo compilador. O compilador junta as informações sobre a operação em pacotes e,
posteriormente, essas informações são usadas para avaliar a operação em tempo de execução.
Como parte do processo, as variáveis do tipo dynamic são compiladas em variáveis do tipo
object . Portanto, o tipo dynamic existe somente em tempo de compilação e não em tempo de
execução.
O exemplo a seguir compara uma variável do tipo dynamic a uma variável do tipo object .
Para verificar o tipo de cada variável no tempo de compilação, coloque o ponteiro do mouse
sobre dyn ou obj nas instruções WriteLine . Copie o seguinte código para um editor em que
o IntelliSense está disponível. O IntelliSense mostra dinâmico para dyn e objeto para obj .
class Program
{
static void Main(string[] args)
{
dynamic dyn = 1;
object obj = 1;
// Rest the mouse pointer over dyn and obj to see their
// types at compile time.
System.Console.WriteLine(dyn.GetType());
System.Console.WriteLine(obj.GetType());
}
}
As instruções WriteLine exibem os tipos de tempo de execução de dyn e obj . Nesse ponto,
ambos têm o mesmo tipo, inteiro. A saída a seguir será produzida:
System.Int32
System.Int32
Para ver a diferença entre dyn e obj em tempo de compilação, adicione as duas linhas a
seguir entre as declarações e as instruções WriteLine no exemplo anterior.
dyn = dyn + 3;
obj = obj + 3;
O exemplo a seguir usa dynamic em várias declarações. O método Main também compara a
verificação de tipo em tempo de compilação com a verificação de tipo em tempo de execução.
using System;
namespace DynamicExamples
{
class Program
{
static void Main(string[] args)
{
ExampleClass ec = new ExampleClass();
Console.WriteLine(ec.exampleMethod(10));
Console.WriteLine(ec.exampleMethod("value"));
class ExampleClass
{
static dynamic field;
dynamic prop { get; set; }
if (d is int)
{
return local;
}
else
{
return two;
}
}
}
}
// Results:
// Local variable
// 2
// Local variable
Consulte também
Referência de C#
Palavras-chave do C#
Eventos
Usando o tipo dynamic
Melhores práticas para o uso de cadeias de caracteres
Operações básicas de cadeias de caracteres
Criando novas cadeias de caracteres
Operadores cast e teste de tipo
Como converter com segurança usando correspondência de padrões e os operadores as e
is
Passo a passo: Criando e usando objetos dinâmicos
System.Object
System.String
System.Dynamic.DynamicObject
class (Referência de C#)
23/10/2019 • 4 minutes to read • Edit Online
class TestClass
{
// Methods, properties, fields, events, delegates
// and nested classes go here.
}
Comentários
Somente a herança única é permitida em C#. Em outras palavras, uma classe pode herdar a
implementação de apenas uma classe base. No entanto, uma classe pode implementar mais
de uma interface. A tabela a seguir mostra exemplos de implementação de interface e
herança de classe:
HERANÇA EXEMPLO
Classes que você declara diretamente dentro de um namespace, não aninhadas em outras
classes, podem ser públicas ou internas. As classes são internal por padrão.
Os membros da classe, incluindo classes aninhadas, podem ser públicos, internos
protegidos, protegidos, internos, privados ou protegidos privados. Os membros são
private por padrão.
Exemplo
O exemplo a seguir demonstra a declaração de métodos, construtores e campos de classe.
Ele também demonstra a instanciação de objetos e a impressão de dados de instância. Neste
exemplo, duas classes são declaradas. A primeira classe, Child , contém dois campos
particulares ( name e age ), dois construtores públicos e um método público. A segunda
classe, StringTest , é usada para conter Main .
class Child
{
private int age;
private string name;
// Default constructor:
public Child()
{
name = "N/A";
}
// Constructor:
public Child(string name, int age)
{
this.name = name;
this.age = age;
}
// Printing method:
public void PrintChild()
{
Console.WriteLine("{0}, {1} years old.", name, age);
}
}
class StringTest
{
static void Main()
{
// Create objects by using the new operator:
Child child1 = new Child("Craig", 11);
Child child2 = new Child("Sally", 10);
// Display results:
Console.Write("Child #1: ");
child1.PrintChild();
Console.Write("Child #2: ");
child2.PrintChild();
Console.Write("Child #3: ");
child3.PrintChild();
}
}
/* Output:
Child #1: Craig, 11 years old.
Child #2: Sally, 10 years old.
Child #3: N/A, 0 years old.
*/
Comentários
Observe que, no exemplo anterior, os campos particulares ( name e age ) só podem ser
acessados por meio dos métodos públicos da classe Child . Por exemplo, você não pode
imprimir o nome do filho, do método Main , usando uma instrução como esta:
Console.Write(child1.name); // Error
Acessar membros particulares de Child de Main seria possível apenas se Main fosse um
membro da classe.
Tipos declarados dentro de uma classe sem um modificador de acesso têm o valor padrão
de private , portanto, os membros de dados neste exemplo ainda seriam private se a
palavra-chave fosse removida.
Por fim, observe que, para o objeto criado usando o construtor sem parâmetro ( child3 ), o
campo age foi inicializado como zero por padrão.
Especificação da linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da
linguagem é a fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Tipos de referência
interface (Referência de C#)
23/10/2019 • 3 minutes to read • Edit Online
Uma interface contém apenas as assinaturas de métodos, propriedades, eventos ou indexadores. Uma
classe ou struct que implementa a interface deve implementar os membros da interface que estão
especificados na definição de interface. No exemplo a seguir, a classe ImplementationClass deve
implementar um método chamado SampleMethod que não tem parâmetros e retorna void .
Para obter mais informações e exemplos, consulte Interfaces.
Exemplo
interface ISampleInterface
{
void SampleMethod();
}
Uma interface pode ser um membro de um namespace ou de uma classe e pode conter assinaturas dos
seguintes membros:
Métodos
Propriedades
Indexadores
Eventos
Uma interface pode herdar de uma ou mais interfaces base.
Quando uma lista de tipos base contém uma classe base e interfaces, a classe base deve vir em primeiro
na lista.
Uma classe que implementa uma interface pode implementar membros dessa interface explicitamente.
Um membro implementado explicitamente não pode ser acessado por meio de uma instância da classe,
mas apenas por meio de uma instância da interface.
Para obter mais detalhes e exemplos de código sobre a implementação explícita da interface, consulte
Implementação explícita da interface.
Exemplo
O exemplo a seguir demonstra a implementação da interface. Neste exemplo, a interface contém a
declaração de propriedade e a classe contém a implementação. Qualquer instância de uma classe que
implementa IPoint tem propriedades de inteiro x e y .
interface IPoint
{
// Property signatures:
int x
{
get;
set;
}
int y
{
get;
set;
}
}
// Constructor:
public Point(int x, int y)
{
_x = x;
_y = y;
}
// Property implementation:
public int x
{
get
{
return _x;
}
set
{
_x = value;
}
}
public int y
{
get
{
return _y;
}
set
{
_y = value;
}
}
}
class MainClass
{
static void PrintPoint(IPoint p)
{
Console.WriteLine("x={0}, y={1}", p.x, p.y);
Console.WriteLine("x={0}, y={1}", p.x, p.y);
}
Especificação da linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é
a fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Tipos de referência
Interfaces
Usando propriedades
Usando indexadores
class
struct
Interfaces
void (Referência de C#)
23/10/2019 • 2 minutes to read • Edit Online
Quando usado como o tipo de retorno para um método, void especifica que o método não retorna um valor.
void não é permitido na lista de parâmetros de um método. Um método que não usa parâmetros e não retorna
nenhum valor é declarado da seguinte maneira:
void também é usado em um contexto desprotegido para declarar um ponteiro para um tipo desconhecido.
Para obter mais informações, consulte Tipos de ponteiros.
void é um alias para o tipo System.Void do .NET Framework.
Especificação da linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a fonte
definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Tipos de referência
Tipos de valor
Métodos
Código não seguro e ponteiros
var (Referência de C#)
23/10/2019 • 2 minutes to read • Edit Online
Começando com o Visual C# 3.0, as variáveis que são declaradas no escopo do método podem ter um
“tipo” implícito var . Uma variável local de tipo implícito é fortemente tipada, como se você mesmo
tivesse declarado o tipo, mas o compilador determina o tipo. As duas declarações a seguir de i são
funcionalmente equivalentes:
Para obter mais informações, consulte Variáveis locais de tipo implícito e Relacionamentos de tipo em
operações de consulta LINQ.
Exemplo
O exemplo a seguir mostra duas expressões de consulta. Na primeira expressão, o uso de var é
permitido, mas não é necessário, pois o tipo do resultado da consulta pode ser declarado explicitamente
como um IEnumerable<string> . No entanto, na segunda expressão, var permite que o resultado seja
uma coleção de tipos anônimos e o nome desse tipo não é acessível, exceto para o próprio compilador.
O uso de var elimina a necessidade de criar uma nova classe para o resultado. Observe que no
exemplo 2, a variável de iteração foreach item também deve ser implicitamente tipada.
Consulte também
Referência de C#
Guia de Programação em C#
Variáveis Locais Tipadas Implicitamente
Tipos não gerenciados (referência em C#)
27/11/2019 • 2 minutes to read • Edit Online
using System;
Uma estrutura genérica pode ser a fonte de tipos construídos não gerenciados e não gerenciados. O exemplo
anterior define um struct genérico Coords<T> e apresenta os exemplos de tipos construídos não gerenciados.
O exemplo de um tipo não gerenciado é Coords<object> . Não é não gerenciado porque tem os campos do tipo
object , que não são gerenciados. Se você quiser que todos os tipos construídos sejam tipos não gerenciados,
use a restrição unmanaged na definição de uma struct genérica:
Consulte também
Referência de C#
Tipos de ponteiro
Tipos relativos a memória e extensão
Operador sizeof
Operador stackalloc
Tabela de tipos internos (Referência de C#)
29/11/2019 • 2 minutes to read • Edit Online
A tabela a seguir mostra as palavras-chave para C# tipos internos, que são aliases de tipos predefinidos no
namespace System:
bool System.Boolean
byte System.Byte
sbyte System.SByte
char System.Char
decimal System.Decimal
double System.Double
float System.Single
int System.Int32
uint System.UInt32
long System.Int64
ulong System.UInt64
object System.Object
short System.Int16
ushort System.UInt16
string System.String
Comentários
Todos os tipos na tabela, exceto object e string , são referidos como tipos simples.
As palavras-chave do tipo C# e seus aliases são intercambiáveis. Por exemplo, é possível declarar uma variável
de inteiro usando uma das seguintes declarações:
int x = 123;
System.Int32 y = 123;
Use o operador typeof para obter a instância System.Type que representa o tipo especificado:
Type stringType = typeof(string);
Console.WriteLine(stringType.FullName);
// Output:
// System.String
// System.Double
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Tipos de valor
Tipos de referência
Tabela de valores padrão
dynamic
Tabela de tipos deC# valor (referência)
29/11/2019 • 2 minutes to read • Edit Online
bool Booliano
enum Enumeração
Consulte também
Referência de C#
Tabela de valores padrão
Tipos de valor
Tabela de formatação de resultados numéricos
Tabela de valores padrão (referência de C#)
29/11/2019 • 2 minutes to read • Edit Online
bool false
Qualquer tipo de valor que permite valor nulo Uma instância para a qual a propriedade HasValue é
false e a propriedade Value não está definida. Esse
valor padrão também é conhecido como o valor nulo de
um tipo de valor anulável.
Use o operador padrão para produzir o valor padrão de um tipo, como mostra o exemplo a seguir:
int a = default(int);
Começando no C# 7.1, você pode usar o default literal para inicializar uma variável com o valor padrão de
seu tipo:
int a = default;
Para um tipo de valor, o construtor implícito sem parâmetros também produz o valor padrão do tipo, como
mostra o seguinte exemplo:
Especificação da linguagem C#
Para obter mais informações, confira as seguintes seções da especificação da linguagem C#:
Valores padrão
Construtores padrão
Consulte também
Referência de C#
Palavras-chave do C#
Tabela de tipos internos
Construtores
Tabela de formatação de resultados numéricos
(Referência de C#)
04/11/2019 • 2 minutes to read • Edit Online
A tabela a seguir mostra os especificadores de formato compatíveis para formatação de resultados numéricos. O
resultado formatado na última coluna corresponde ao "en-US" CultureInfo.
string s = $"{2.5:F0}"; 3
Comentários
Você pode usar um especificador de formato para criar uma cadeia de caracteres de formato. A cadeia de
caracteres de formato é da seguinte forma: Axx , em que
A é o especificador de formato, que controla o tipo de formatação aplicada ao valor numérico.
xx é o especificador de precisão, que afeta o número de dígitos na saída formatada. O valor do especificador
de precisão varia de 0 a 99.
Os especificadores de formato decimal ("D" ou "d") e hexadecimal ("X" ou "x") são compatíveis apenas com tipos
integrais. O especificador de formato de ida e volta ("R" ou "r") é compatível apenas com tipos Single, Double e
BigInteger.
As cadeias de caractere de formato numérico padrão têm suporte de:
Algumas sobrecargas do método ToString de todos os tipos numéricos. Por exemplo, você pode fornecer
uma cadeia de caracteres de formato numérico para os métodos Int32.ToString(String) e
Int32.ToString(String, IFormatProvider).
O recurso de formatação composta do .NET, que é compatível com o método String.Format, por exemplo.
Cadeias de caracteres interpoladas.
Para obter mais informações, confira Cadeias de caracteres de formato numérico padrão.
Consulte também
Referência de C#
Guia de Programação em C#
Formatando tipos
Formatação de composição
Interpolação de cadeia de caracteres
string
Modificadores de acesso (Referência de C#)
04/11/2019 • 2 minutes to read • Edit Online
Os seis seguintes níveis de acessibilidade podem ser especificados usando os modificadores de acesso:
public : o acesso não é restrito.
protected : o acesso é limitado à classe que os contém ou aos tipos derivados da classe que os contém.
internal : o acesso é limitado ao assembly atual.
protected internal : o acesso é limitado ao assembly atual ou aos tipos derivados da classe que os
contém.
private : o acesso é limitado ao tipo recipiente.
private protected : o acesso é limitado à classe que o contém ou a tipos derivados da classe que o
contém no assembly atual.
Esta seção também apresenta o seguinte:
Níveis de acessibilidade: utilização dos quatro modificadores de acesso para declarar seis níveis de
acessibilidade.
Domínio de acessibilidade: especifica em que lugar, nas seções do programa, um membro pode ser
referenciado.
Restrições no uso de níveis de acessibilidade: um resumo das restrições sobre o uso de níveis de
acessibilidade declarados.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Modificadores de acesso
Palavras-chave de acesso
Modificadores
Níveis de acessibilidade (Referência de C#)
23/10/2019 • 3 minutes to read • Edit Online
Use os modificadores de acesso, public , protected , internal ou private , para especificar um dos níveis de
acessibilidade declarada a seguir para membros.
Apenas um modificador de acesso é permitido para um membro ou tipo, exceto quando você usa as
combinações protected internal e private protected .
Os modificadores de acesso não são permitidos em namespaces. Namespaces não têm nenhuma restrição de
acesso.
Dependendo do contexto no qual ocorre uma declaração de membro, apenas algumas acessibilidades
declaradas são permitidas. Se não for especificado nenhum modificador de acesso em uma declaração de
membro, uma acessibilidade padrão será usada.
Os tipos de nível superior, que não estão aninhados em outros tipos, podem ter apenas a acessibilidade
internal ou public . A acessibilidade de padrão para esses tipos é internal .
Tipos aninhados, que são membros de outros tipos, podem ter acessibilidades declaradas conforme indicado
na tabela a seguir.
protected
internal
private
protected internal
private protected
internal
private
Especificação da Linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a
fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Modificadores de acesso
Domínio de acessibilidade
Restrições ao uso de níveis de acessibilidade
Modificadores de acesso
public
private
protected
internal
Domínio de acessibilidade (Referência de C#)
23/10/2019 • 3 minutes to read • Edit Online
O domínio de acessibilidade de um membro especifica em que seções do programa um membro pode ser
referenciado. Se o membro estiver aninhado dentro de outro tipo, seu domínio de acessibilidade será
determinado pelo nível de acessibilidade do membro e pelo domínio de acessibilidade do tipo imediatamente
contido.
O domínio de acessibilidade de um tipo de nível superior é, pelo menos, o texto do programa do projeto no qual
ele é declarado. Isto é, o domínio inclui todos os arquivos de origem deste projeto. O domínio de acessibilidade
de um tipo aninhado é, pelo menos, o texto do programa do tipo no qual ele é declarado. Isto é, o domínio é o
corpo do tipo, que inclui todos os tipos aninhados. O domínio de acessibilidade de um tipo aninhado nunca
excede o do tipo contido. Esses conceitos são demonstrados no exemplo a seguir.
Exemplo
Este exemplo contém um tipo de nível superior, T1 e duas classes aninhadas, M1 e M2 . As classes contêm
campos que têm diferentes acessibilidades declaradas. No método Main , um comentário segue cada instrução
para indicar o domínio de acessibilidade de cada membro. Observe que as instruções que tentam fazer referência
aos membros inacessíveis são tornadas inertes ao serem transformadas em comentários. Se você quiser ver os
erros de compilador causados ao referenciar um membro inacessível, remova os comentários um por vez.
public class T1
{
public static int publicInt;
internal static int internalInt;
private static int privateInt = 0;
static T1()
{
// T1 can access public or internal members
// in a public or private (or internal) nested class.
M1.publicInt = 1;
M1.internalInt = 2;
M2.publicInt = 3;
M2.internalInt = 4;
public class M1
{
public static int publicInt;
internal static int internalInt;
private static int privateInt = 0;
}
private class M2
{
public static int publicInt = 0;
internal static int internalInt = 0;
private static int privateInt = 0;
}
}
class MainClass
{
{
static void Main()
{
// Access is unlimited.
T1.publicInt = 1;
// Access is unlimited.
T1.M1.publicInt = 1;
Especificação da Linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a fonte
definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Modificadores de acesso
Níveis de acessibilidade
Restrições ao uso de níveis de acessibilidade
Modificadores de acesso
public
private
protected
internal
Restrições ao uso de níveis de acessibilidade
(Referência em C#)
23/10/2019 • 4 minutes to read • Edit Online
Quando você especifica um tipo em uma declaração, verifique se o nível de acessibilidade do tipo é dependente
do nível de acessibilidade de um membro ou de outro tipo. Por exemplo, a classe base direta deve ser, pelo menos,
tão acessível quanto a classe derivada. As seguintes declarações causam um erro do compilador porque a classe
base BaseClass é menos acessível que a MyClass :
CONTEX TO COMENTÁRIOS
Constantes O tipo de uma constante deve ser, pelo menos, tão acessível
quanto a própria constante.
Exemplo
O exemplo a seguir contém declarações incorretas de tipos diferentes. O comentário que segue cada declaração
indica o erro do compilador esperado.
// Restrictions on Using Accessibility Levels
// CS0052 expected as well as CS0053, CS0056, and CS0057
// To make the program work, change access level of both class B
// and MyPrivateMethod() to public.
using System;
// A delegate:
delegate int MyDelegate();
class B
{
// A private method:
static int MyPrivateMethod()
{
return 0;
}
}
public class A
{
// Error: The type B is less accessible than the field A.myField.
public B myField = new B();
public B MyMethod()
{
// Error: The type B is less accessible
// than the method A.MyMethod.
return new B();
}
Especificação da linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a fonte
definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Modificadores de acesso
Domínio de acessibilidade
Níveis de acessibilidade
Modificadores de acesso
public
private
protected
internal
internal (Referência de C#)
04/11/2019 • 3 minutes to read • Edit Online
Esta página aborda o acesso internal . A palavra-chave internal também faz parte do modificador
de acesso protected internal .
Tipos ou membros internos são acessíveis somente em arquivos no mesmo assembly, como neste
exemplo:
Para obter uma comparação de internal com os outros modificadores de acesso, consulte Níveis de
acessibilidade e Modificadores de acesso.
Para saber mais sobre assemblies, confira Assembly no .NET.
Um uso comum do acesso interno é no desenvolvimento baseado em componente, porque ele permite
que um grupo de componentes colabore de maneira particular, sem serem expostos para o restante do
código do aplicativo. Por exemplo, uma estrutura para a criação de interfaces gráficas do usuário pode
fornecer as classes Control e Form , que cooperam através do uso de membros com acesso interno. Uma
vez que esses membros são internos, eles não são expostos ao código que está usando a estrutura.
É um erro fazer referência a um tipo ou um membro com acesso interno fora do assembly no qual ele foi
definido.
Exemplo
Este exemplo contém dois arquivos, Assembly1.cs e Assembly1_a.cs . O primeiro arquivo contém uma
classe base interna BaseClass . No segundo arquivo, uma tentativa de instanciar a BaseClass produzirá
um erro.
// Assembly1.cs
// Compile with: /target:library
internal class BaseClass
{
public static int intM = 0;
}
// Assembly1_a.cs
// Compile with: /reference:Assembly1.dll
class TestAccess
{
static void Main()
{
var myBase = new BaseClass(); // CS0122
}
}
Exemplo
Neste exemplo, use os mesmos arquivos que você usou no exemplo 1 e altere o nível de acessibilidade da
BaseClass para public . Também altere o nível de acessibilidade do membro intM para internal .
Nesse caso, você pode instanciar a classe, mas não pode acessar o membro interno.
// Assembly2.cs
// Compile with: /target:library
public class BaseClass
{
internal static int intM = 0;
}
// Assembly2_a.cs
// Compile with: /reference:Assembly2.dll
public class TestAccess
{
static void Main()
{
var myBase = new BaseClass(); // Ok.
BaseClass.intM = 444; // CS0117
}
}
Especificação da Linguagem C#
Para obter mais informações, veja Acessibilidade declarada na Especificação da Linguagem C#. A
especificação da linguagem é a fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Modificadores de acesso
Níveis de acessibilidade
Modificadores
public
private
protected
private (Referência de C#)
04/11/2019 • 2 minutes to read • Edit Online
Esta página aborda o acesso private . A palavra-chave private também faz parte do modificador
de acesso private protected .
Acesso particular é o nível de acesso menos permissivo. Membros particulares são acessíveis somente
dentro do corpo da classe ou do struct em que são declarados, como neste exemplo:
class Employee
{
private int i;
double d; // private access by default
}
Exemplo
Neste exemplo, a classe Employee contém dois membros de dados particulares, name e salary .
Como membros particulares, eles não podem ser acessados, exceto por métodos de membro.
Métodos públicos chamados GetName e Salary foram adicionados para permitir acesso controlado
aos membros particulares. O membro name é acessado por meio de um método público e o membro
salary é acessado por meio de uma propriedade somente leitura pública. ( Consulte Propriedades
para obter mais informações.)
class Employee2
{
private string name = "FirstName, LastName";
private double salary = 100.0;
class PrivateTest
{
static void Main()
{
var e = new Employee2();
Especificação da linguagem C#
Para obter mais informações, veja Acessibilidade declarada na Especificação da Linguagem C#. A
especificação da linguagem é a fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Modificadores de acesso
Níveis de acessibilidade
Modificadores
public
protected
internal
protected (Referência de C#)
04/11/2019 • 2 minutes to read • Edit Online
Esta página aborda o acesso protected . A palavra-chave protected também faz parte dos
modificadores de acesso protected internal e private protected .
Um membro protegido é acessível dentro de sua classe e por instâncias da classe derivada.
Para obter uma comparação de protected com os outros modificadores de acesso, consulte Níveis de
acessibilidade.
Exemplo
Um membro protegido de uma classe base será acessível em uma classe derivada somente se o acesso
ocorrer por meio do tipo da classe derivada. Por exemplo, considere o seguinte segmento de código:
class A
{
protected int x = 123;
}
class B : A
{
static void Main()
{
var a = new A();
var b = new B();
A instrução a.x = 10 gera um erro porque ela é feita dentro do método estático Main e não em uma
instância da classe B.
Membros de struct não podem ser protegidos porque o struct não pode ser herdado.
Exemplo
Nesse exemplo, a classe DerivedPoint é derivada de Point . Portanto, você pode acessar os membros
protegidos da classe base diretamente da classe derivada.
class Point
{
protected int x;
protected int y;
}
Se você alterar os níveis de acesso de x e y para private, o compilador emitirá as mensagens de erro:
'Point.y' is inaccessible due to its protection level.
Especificação da linguagem C#
Para obter mais informações, veja Acessibilidade declarada na Especificação da Linguagem C#. A
especificação da linguagem é a fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Modificadores de acesso
Níveis de acessibilidade
Modificadores
public
private
internal
Questões de segurança de palavras-chave virtuais internas
public (Referência de C#)
04/11/2019 • 2 minutes to read • Edit Online
A palavra-chave public é um modificador de acesso para tipos e membros de tipo. Acesso público é
o nível de acesso mais permissivo. Não há nenhuma restrição quanto ao acesso a membros públicos,
como neste exemplo:
class SampleClass
{
public int x; // No access restrictions.
}
Exemplo
No exemplo a seguir, duas classes são declaradas, PointTest e MainClass . Os membros públicos x
e y de PointTest são acessados diretamente de MainClass .
class PointTest
{
public int x;
public int y;
}
class MainClass4
{
static void Main()
{
var p = new PointTest();
// Direct access to public members.
p.x = 10;
p.y = 15;
Console.WriteLine($"x = {p.x}, y = {p,y}");
}
}
// Output: x = 10, y = 15
Se alterar o nível de acesso de public para particular ou protegido, você receberá a mensagem de
erro:
'PointTest.y' é inacessível devido ao seu nível de proteção.
Especificação da linguagem C#
Para obter mais informações, veja Acessibilidade declarada na Especificação da Linguagem C#. A
especificação da linguagem é a fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
Modificadores de acesso
Palavras-chave do C#
Modificadores de acesso
Níveis de acessibilidade
Modificadores
private
protected
internal
protected internal (referência do C#)
04/11/2019 • 2 minutes to read • Edit Online
Exemplo
Um membro protegido interno de uma classe base é acessível em qualquer tipo em seu assembly recipiente.
Ele também é acessível em uma classe derivada localizada em outro assembly somente se o acesso ocorre
por meio de uma variável do tipo de classe derivada. Por exemplo, considere o seguinte segmento de código:
// Assembly1.cs
// Compile with: /target:library
public class BaseClass
{
protected internal int myValue = 0;
}
class TestAccess
{
void Access()
{
var baseObject = new BaseClass();
baseObject.myValue = 5;
}
}
// Assembly2.cs
// Compile with: /reference:Assembly1.dll
class DerivedClass : BaseClass
{
static void Main()
{
var baseObject = new BaseClass();
var derivedObject = new DerivedClass();
Este exemplo contém dois arquivos, Assembly1.cs e Assembly2.cs . O primeiro arquivo contém uma classe
base pública, BaseClass , e outra classe, TestAccess . BaseClass tem um membro interno protegido,
myValue , que é acessado pelo tipo TestAccess . No segundo arquivo, uma tentativa de acessar myValue por
meio de uma instância de BaseClass produzirá um erro, enquanto um acesso a esse membro por meio de
uma instância de uma classe derivada, DerivedClass , terá êxito.
Membros de struct não podem ser protected internal porque o struct não pode ser herdado.
Especificação da linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a
fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Modificadores de acesso
Níveis de acessibilidade
Modificadores
public
private
internal
Questões de segurança de palavras-chave virtuais internas
private protected (referência do C#)
04/11/2019 • 3 minutes to read • Edit Online
NOTE
O modificador de acesso private protected é válido no C# versão 7.2 e posterior.
Exemplo
Um membro particular protegido de uma classe base é acessível de tipos derivados em seu assembly
recipiente apenas se o tipo estático da variável é o tipo da classe derivada. Por exemplo, considere o seguinte
segmento de código:
// Assembly1.cs
// Compile with: /target:library
public class BaseClass
{
private protected int myValue = 0;
}
// Assembly2.cs
// Compile with: /reference:Assembly1.dll
class DerivedClass2 : BaseClass
{
void Access()
{
// Error CS0122, because myValue can only be
// accessed by types in Assembly1
// myValue = 10;
}
}
Este exemplo contém dois arquivos, Assembly1.cs e Assembly2.cs . O primeiro arquivo contém uma classe
base pública, BaseClass , e um tipo derivado dela, DerivedClass1 . BaseClass tem um membro particular
protegido, myValue , que DerivedClass1 tenta acessar de duas maneiras. A primeira tentativa de acessar
myValue por meio de uma instância de BaseClass produzirá um erro. No entanto, a tentativa de usá-lo
como um membro herdado em DerivedClass1 terá êxito. No segundo arquivo, uma tentativa de acessar
myValue como um membro herdado de DerivedClass2 produzirá um erro, pois ele é acessível apenas por
tipos derivados em Assembly1.
Membros de struct não podem ser private protected porque o struct não pode ser herdado.
Especificação da linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a
fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Modificadores de acesso
Níveis de acessibilidade
Modificadores
public
private
internal
Questões de segurança de palavras-chave virtuais internas
abstract (Referência de C#)
04/11/2019 • 5 minutes to read • Edit Online
O modificador abstract indica que o item que está sendo modificado tem uma implementação ausente
ou incompleta. O modificador abstrato pode ser usado com classes, métodos, propriedades,
indexadores e eventos. Use o modificador abstract em uma declaração de classe para indicar que uma
classe se destina somente a ser uma classe base de outras classes, não instanciada por conta própria.
Membros marcados como abstratos precisam ser implementados por classes não abstratas que
derivam da classe abstrata.
Exemplo
Neste exemplo, a classe Square deve fornecer uma implementação de GetArea porque deriva de
Shape :
A implementação é fornecida por uma substituição de método, que é um membro de uma classe
não abstrata.
É um erro usar os modificadores static ou virtual em uma declaração de método abstrato.
Propriedades abstratas se comportam como métodos abstratos, exceto pelas diferenças na sintaxe de
declaração e chamada.
É um erro usar o modificador abstract em uma propriedade estática.
Uma propriedade herdada abstrata pode ser substituída em uma classe derivada incluindo uma
declaração de propriedade que usa o modificador override.
Para obter mais informações sobre classes abstratas, consulte Classes e membros de classes abstratos e
lacrados.
Uma classe abstrata deve fornecer uma implementação para todos os membros de interface.
Uma classe abstrata que implementa uma interface pode mapear os métodos de interface em métodos
abstratos. Por exemplo:
interface I
{
void M();
}
abstract class C : I
{
public abstract void M();
}
Exemplo
Nesse exemplo, a classe DerivedClass é derivada de uma classe abstrata BaseClass . A classe abstrata
contém um método abstrato, AbstractMethod e duas propriedades abstratas, X e Y .
abstract class BaseClass // Abstract class
{
protected int _x = 100;
protected int _y = 150;
public abstract void AbstractMethod(); // Abstract method
public abstract int X { get; }
public abstract int Y { get; }
}
No exemplo anterior, se você tentar instanciar a classe abstrata usando uma instrução como esta:
Você receberá uma mensagem de erro informando que o compilador não pode criar uma instância da
classe abstrata "BaseClass".
Especificação da Linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é
a fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
Modificadores
virtual
override
Palavras-chave do C#
async (Referência de C#)
23/10/2019 • 8 minutes to read • Edit Online
Use o modificador async para especificar que um método, uma expressão lambda ou um método
anônimo é assíncrono. Se você usar esse modificador em um método ou expressão, ele será referido
como um método assíncrono. O exemplo a seguir define um método assíncrono chamado
ExampleMethodAsync :
Se você é iniciante em programação assíncrona ou não compreende como um método assíncrono usa o
operador await para fazer trabalhos potencialmente longos sem bloquear o thread do chamador, leia a
introdução em Programação assíncrona com async e await. O código a seguir é encontrado dentro de
um método assíncrono e chama o método HttpClient.GetStringAsync:
Um método assíncrono será executado de forma síncrona até atingir sua primeira expressão await e,
nesse ponto, ele será suspenso até que a tarefa aguardada seja concluída. Enquanto isso, o controle será
retornado ao chamador do método, como exibido no exemplo da próxima seção.
Se o método que a palavra-chave async modifica não contiver uma expressão ou instrução await , ele
será executado de forma síncrona. Um aviso do compilador o alertará sobre quaisquer métodos
assíncronos que não contenham instruções await , pois essa situação poderá indicar um erro. Consulte
Aviso do compilador (nível 1) CS4014.
A palavra-chave async é contextual, pois ela será uma palavra-chave somente quando modificar um
método, uma expressão lambda ou um método anônimo. Em todos os outros contextos, ela será
interpretada como um identificador.
Exemplo
O exemplo a seguir mostra a estrutura e o fluxo de controle entre um manipulador de eventos
assíncronos, StartButton_Click e um método assíncrono, ExampleMethodAsync . O resultado do método
assíncrono é o número de caracteres de uma página da Web. O código será adequado para um
aplicativo do WPF (Windows Presentation Foundation) ou da Windows Store que você criar no Visual
Studio, consulte os comentários do código para configurar o aplicativo.
Você pode executar esse código no Visual Studio como um aplicativo do WPF (Windows Presentation
Foundation) ou um aplicativo da Windows Store. Você precisa de um controle de botão chamado
StartButton e de um controle de caixa de texto chamado ResultsTextBox . Lembre-se de definir os
nomes e o manipulador para que você tenha algo assim:
<Button Content="Button" HorizontalAlignment="Left" Margin="88,77,0,0" VerticalAlignment="Top"
Width="75"
Click="StartButton_Click" Name="StartButton"/>
<TextBox HorizontalAlignment="Left" Height="137" Margin="88,140,0,0" TextWrapping="Wrap"
Text="<Enter a URL>" VerticalAlignment="Top" Width="310" Name="ResultsTextBox"/>
try
{
int length = await ExampleMethodAsync();
// Note that you could put "await ExampleMethodAsync()" in the next line where
// "length" is, but due to when '+=' fetches the value of ResultsTextBox, you
// would not see the global side effect of ExampleMethodAsync setting the text.
ResultsTextBox.Text += String.Format("Length: {0:N0}\n", length);
}
catch (Exception)
{
// Process the exception if one occurs.
}
}
IMPORTANT
Para obter mais informações sobre tarefas e o código que é executado enquanto aguarda uma tarefa, consulte
Programação assíncrona com async e await. Para obter um exemplo completo do WPF que use elementos
semelhantes, confira Instruções passo a passo: acessar a Web usando Async e Await.
Tipos de Retorno
Um método assíncrono pode conter os seguintes tipos de retorno:
Task
Task<TResult>
void. Os métodos async void geralmente são desencorajados para código que não são
manipuladores de eventos, porque os chamadores não podem await esses métodos e devem
implementar um mecanismo diferente para relatar a conclusão bem-sucedida ou condições de erro.
Começando com o C# 7.0, qualquer tipo que tenha um método acessível GetAwaiter . O tipo
System.Threading.Tasks.ValueTask<TResult> é um exemplo de uma implementação assim. Ele está
disponível ao adicionar o pacote NuGet System.Threading.Tasks.Extensions .
O método assíncrono não pode declarar os parâmetros in, ref nem out e também não pode ter um valor
retornado por referência, mas pode chamar métodos que tenham esses parâmetros.
Você especificará Task<TResult> como o tipo de retorno de um método assíncrono se a instrução return
do método especificar um operando do tipo TResult . Você usará Task se nenhum valor significativo
for retornado quando o método for concluído. Isto é, uma chamada ao método retorna uma Task , mas
quando a Task for concluída, qualquer expressão await que esteja aguardando a Task será avaliada
como void .
O tipo de retorno void é usado principalmente para definir manipuladores de eventos que exigem esse
tipo de retorno. O chamador de um método assíncrono de retorno void não pode aguardá-lo e
capturar exceções acionadas pelo método.
A partir do C# 7.0, você retorna outro tipo, geralmente um tipo de valor, que tenha um método
GetAwaiter para minimizar as alocações de memória nas seções do código críticas ao desempenho.
Consulte também
AsyncStateMachineAttribute
await
Passo a passo: acessar a Web usando Async e Await
Programação assíncrona com async e await
const (Referência de C#)
04/11/2019 • 4 minutes to read • Edit Online
Use a palavra-chave const para declarar um campo constante ou um local constante. Campos e locais
constantes não são variáveis e não podem ser modificados. As constantes podem ser números, valores
boolianos, cadeias de caracteres ou uma referência nula. Não crie uma constante para representar informações
que você espera mudar a qualquer momento. Por exemplo, não use um campo constante para armazenar o
preço de um serviço, um número de versão de produto ou a marca de uma empresa. Esses valores podem
mudar ao longo do tempo e como os compiladores propagam constantes, outro código compilado com as
bibliotecas terá que ser recompilado para ver as alterações. Veja também a palavra-chave readonly. Por exemplo:
const int X = 0;
public const double GravitationalConstant = 6.673e-11;
private const string ProductName = "Visual C#";
Comentários
O tipo de uma declaração constante especifica o tipo dos membros que a declaração apresenta. O inicializador
de um local ou um campo constante deve ser uma expressão constante que pode ser implicitamente convertida
para o tipo de destino.
Uma expressão constante é uma expressão que pode ser completamente avaliada em tempo de compilação.
Portanto, os únicos valores possíveis para as constantes de tipos de referência são string e uma referência
nula.
A declaração constante pode declarar constantes múltiplas como:
NOTE
A palavra-chave readonly é diferente da palavra-chave const . O campo const pode ser inicializado apenas na
declaração do campo. Um campo readonly pode ser inicializado na declaração ou em um construtor. Portanto, campos
readonly podem ter valores diferentes dependendo do construtor usado. Além disso, embora um campo const seja
uma constante em tempo de compilação, o campo readonly pode ser usado para constantes de tempo de execução,
como nesta linha: public static readonly uint l1 = (uint)DateTime.Now.Ticks;
Exemplo
public class ConstTest
{
class SampleClass
{
public int x;
public int y;
public const int C1 = 5;
public const int C2 = C1 + 5;
Exemplo
Este exemplo demonstra como usar constantes como variáveis locais.
Especificação da linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a fonte
definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Modificadores
readonly
event (Referência de C#)
04/11/2019 • 4 minutes to read • Edit Online
Exemplo
O exemplo a seguir mostra como declarar e acionar um evento que usa o EventHandler como o tipo delegado
subjacente. Para obter o exemplo de código completo, que também mostra como usar o tipo delegado
EventHandler<TEventArgs> genérico e como assinar um evento e criar um método de manipulador de
eventos, consulte Como publicar eventos em conformidade com as diretrizes do .NET Framework.
Os eventos são um tipo especial de delegado multicast que só podem ser invocados de dentro da classe ou
struct em que eles são declarados (a classe publicadora). Se outras classes ou structs assinarem o evento, seus
respectivos métodos de manipulador de eventos serão chamados quando a classe publicadora acionar o
evento. Para obter mais informações e exemplos de código, consulte Eventos e Delegados.
Os eventos podem ser marcados como public, private, protected, internal, protected internal ou private
protected. Esses modificadores de acesso definem como os usuários da classe podem acessar o evento. Para
obter mais informações, consulte Modificadores de acesso.
Palavras-chave e eventos
As palavras-chave a seguir aplicam-se a eventos.
Um evento pode ser declarado como um evento estático, usando apalavra-chave static. Isso torna o evento
disponível para chamadores a qualquer momento, mesmo se não existir nenhuma instância da classe. Para
obter mais informações, consulte Classes Estáticas e Membros de Classes Estáticas.
Um evento pode ser marcado como um evento virtual, usando a palavra-chave virtual. Isso habilita as classes
derivadas a substituírem o comportamento do evento, usando a palavra-chave override. Para obter mais
informações, consulte Herança. Um evento que substitui um evento virtual também pode ser sealed, o que
especifica que ele não é mais virtual para classes derivadas. Por fim, um evento pode ser declarado abstract, o
que significa que o compilador não gerará os blocos de acessador de evento add e remove . Portanto, classes
derivadas devem fornecer sua própria implementação.
Especificação da Linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a fonte
definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
add
remove
Modificadores
Como combinar delegados (delegados multicast)
extern (Referência de C#)
04/11/2019 • 4 minutes to read • Edit Online
O modificador extern é usado para declarar um método implementado externamente. Um uso comum do
modificador extern é com o atributo DllImport quando você está usando serviços Interop para chamar código
não gerenciado. Nesse caso, o método também deve ser declarado como static conforme mostrado no
seguinte exemplo:
[DllImport("avifil32.dll")]
private static extern void AVIFileInit();
A palavra-chave extern também pode definir um alias de assembly externo que possibilita referenciar diferentes
versões do mesmo componente de dentro de um único assembly. Para obter mais informações, consulte alias
externo.
É um erro usar os modificadores abstract e extern juntos para modificar o mesmo membro. Usar o modificador
extern significa que esse método é implementado fora do código C#, enquanto que usar o modificador
abstract significa que a implementação do método não é fornecida na classe.
A palavra-chave extern possui utilizações mais limitadas em C# do que em C++. Para comparar a palavra-chave
de C# com a palavra-chave de C++, consulte Usando extern para especificar vínculos na referência da linguagem
C++.
Exemplo 1
Neste exemplo, o programa recebe uma cadeia de caracteres do usuário e a exibe dentro de uma caixa de
mensagem. O programa usa o método MessageBox importado da biblioteca User32.dll.
//using System.Runtime.InteropServices;
class ExternTest
{
[DllImport("User32.dll", CharSet=CharSet.Unicode)]
public static extern int MessageBox(IntPtr h, string m, string c, int type);
Exemplo 2
Este exemplo ilustra um programa C# que chama uma biblioteca em C (uma DLL nativa).
1. Crie o seguinte arquivo em C e atribua o nome cmdll.c :
// cmdll.c
// Compile with: -LD
int __declspec(dllexport) SampleMethod(int i)
{
return i*10;
}
2. Abra uma janela do Prompt de Comando de Ferramentas Nativas do Visual Studio x64 (ou x32) do
diretório de instalação do Visual Studio e compile o arquivo cmdll.c digitando cl -LD cmdll.c no prompt
de comando.
3. No mesmo diretório, crie o seguinte arquivo em C# e atribua o nome cm.cs :
// cm.cs
using System;
using System.Runtime.InteropServices;
public class MainClass
{
[DllImport("Cmdll.dll")]
public static extern int SampleMethod(int x);
4. Abra uma janela do Prompt de Comando de Ferramentas Nativas do Visual Studio x64 (ou x32) do
diretório de instalação do Visual Studio e compile o arquivo cm.cs ao digitar:
csc cm.cs (para o prompt de comando do x64) – ou – csc -platform:x86 cm.cs (para o prompt de
comando do x32)
Especificação da linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a fonte
definitiva para a sintaxe e o uso de C#.
Consulte também
System.Runtime.InteropServices.DllImportAttribute
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Modificadores
in (modificador genérico) (Referência de C#)
04/11/2019 • 3 minutes to read • Edit Online
Para parâmetros de tipo genérico, a palavra-chave in especifica que o parâmetro de tipo é contravariante. Você
pode usar a palavra-chave in em delegados e interfaces genéricas.
A contravariância permite que você use um tipo menos derivado do que aquele especificado pelo parâmetro
genérico. Isso permite a conversão implícita de classes que implementam interfaces contravariantes e a conversão
implícita de tipos delegados. A covariância e a contravariância em parâmetros de tipo genérico têm suporte para
tipos de referência, mas não para tipos de valor.
Um tipo pode ser declarado como contravariante em uma interface genérica ou como delegado somente se ele
define o tipo de parâmetros de um método e não o tipo de retorno de um método. Os parâmetros In , ref e
out precisam ser invariáveis, ou seja, nem covariantes nem contravariantes.
Uma interface que tem um parâmetro de tipo contravariante permite que os seus métodos aceitem argumentos de
tipos menos derivados que aqueles especificados pelo parâmetro de tipo de interface. Por exemplo, na interface
IComparer<T>, o tipo T é contravariante e você pode atribuir um objeto do tipo IComparer<Person> a um objeto
do tipo IComparer<Employee> , sem usar nenhum método de conversão especial caso Employee herde Person .
Um delegado contravariante pode ser atribuído a outro delegado do mesmo tipo, mas com um parâmetro de tipo
genérico menos derivado.
Para obter mais informações, consulte Covariância e contravariância.
// Contravariant interface.
interface IContravariant<in A> { }
class Program
{
static void Test()
{
IContravariant<Object> iobj = new Sample<Object>();
IContravariant<String> istr = new Sample<String>();
// Contravariant delegate.
public delegate void DContravariant<in A>(A argument);
Especificação da linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a fonte
definitiva para a sintaxe e o uso de C#.
Consulte também
out
Covariância e Contravariância
Modificadores
Modificador new (referência em C#)
04/11/2019 • 6 minutes to read • Edit Online
Quando usada como um modificador de declaração, a palavra-chave new oculta explicitamente um membro
herdado de uma classe base. Quando você oculta um membro herdado, a versão derivada do membro
substitui a versão da classe base. Embora possa ocultar membros sem usar o modificador new , você obtém
um aviso do compilador. Se você usar new para ocultar explicitamente um membro, ele suprime o aviso.
Você também pode usar a palavra-chave new para criar uma instância de um tipo ou como uma restrição de
tipo genérico.
Para ocultar um membro herdado, declare-o na classe derivada usando o mesmo nome de membro e
modifique-o com a palavra-chave new . Por exemplo:
Neste exemplo, BaseC.Invoke é ocultado por DerivedC.Invoke . O campo x não é afetado porque não é
ocultado por um nome semelhante.
A ocultação de nome por meio da herança assume uma das seguintes formas:
Normalmente, uma constante, campo, propriedade ou tipo que é apresentado em uma classe ou
struct oculta todos os membros da classe base que compartilham seu nome. Há casos especiais. Por
exemplo, se você declarar que um novo campo com o nome N tem um tipo que não pode ser
invocado e um tipo base declarar que N é um método, o novo campo não ocultará a declaração base
na sintaxe de invocação. Para obter mais informações, consulte a seção Pesquisa de membro na
especificação da linguagem C#.
Um método introduzido em uma classe ou struct oculta propriedades, campos e tipos que
compartilham esse nome na classe base. Ele também oculta todos os métodos da classe base que têm
a mesma assinatura.
Um indexador introduzido em uma classe ou struct oculta todos os indexadores de classe base que
têm a mesma assinatura.
É um erro usar new e override no mesmo membro, porque os dois modificadores têm significados
mutuamente exclusivos. O modificador new cria um novo membro com o mesmo nome e faz com que o
membro original seja ocultado. O modificador override estende a implementação de um membro herdado.
Usar o modificador new em uma declaração que não oculta um membro herdado gera um aviso.
Exemplo
Neste exemplo, uma classe base, BaseC e uma classe derivada, DerivedC , usam o mesmo nome do campo
x , que oculta o valor do campo herdado. O exemplo demonstra o uso do modificador new . Ele também
demonstra como acessar os membros ocultos da classe base usando seus nomes totalmente qualificados.
Exemplo
Neste exemplo, uma classe aninhada oculta uma classe que tem o mesmo nome na classe base. O exemplo
demonstra como usar o modificador new para eliminar a mensagem de aviso e como acessar os membros
da classe oculta usando seus nomes totalmente qualificados.
public class BaseC
{
public class NestedC
{
public int x = 200;
public int y;
}
}
Console.WriteLine(c1.x);
Console.WriteLine(c2.x);
}
}
/*
Output:
100
200
*/
Se você remover o modificador new , o programa ainda será compilado e executado, mas você receberá o
seguinte aviso:
The keyword new is required on 'MyDerivedC.x' because it hides inherited member 'MyBaseC.x'.
Especificação da linguagem C#
Para obter mais informações, consulte a seção O modificador new na especificação da linguagem C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Modificadores
Controle de versão com as palavras-chave override e new
Quando usar as palavras-chave override e new
out (modificador genérico) (Referência de C#)
04/11/2019 • 4 minutes to read • Edit Online
Para parâmetros de tipo genérico, a palavra-chave out especifica que o parâmetro de tipo é covariante. Você
pode usar a palavra-chave out em delegados e interfaces genéricas.
A covariância permite que você use um tipo mais derivado do que aquele especificado pelo parâmetro genérico.
Isso permite a conversão implícita de classes que implementam interfaces covariantes e a conversão implícita de
tipos delegados. A covariância e a contravariância têm suporte para tipos de referência, mas não para tipos de
valor.
Uma interface que tem um parâmetro de tipo covariante permite que seus métodos retornem tipos mais
derivados do que aqueles especificados pelo parâmetro de tipo. Por exemplo, já que no .NET Framework 4, em
IEnumerable<T>, o tipo T é covariante, você pode atribuir um objeto do tipo IEnumerable(Of String) a um objeto
do tipo IEnumerable(Of Object) sem usar nenhum método de conversão especial.
Pode ser atribuído a um delegado covariante outro delegado do mesmo tipo, mas com um parâmetro de tipo
genérico mais derivado.
Para obter mais informações, consulte Covariância e contravariância.
// Covariant interface.
interface ICovariant<out R> { }
class Program
{
static void Test()
{
ICovariant<Object> iobj = new Sample<Object>();
ICovariant<String> istr = new Sample<String>();
Em uma interface genérica, um parâmetro de tipo pode ser declarado covariante se ele satisfizer as condições a
seguir:
O parâmetro de tipo é usado apenas como um tipo de retorno dos métodos de interface e não é usado
como um tipo de argumentos de método.
NOTE
Há uma exceção a essa regra. Se, em uma interface covariante, você tiver um delegado genérico contravariante como
um parâmetro de método, você poderá usar o tipo covariante como um parâmetro de tipo genérico para o
delegado. Para obter mais informações sobre delegados genéricos covariantes e contravariantes, consulte Variância
em delegados e Usando variância para delegados genéricos Func e Action.
O parâmetro de tipo não é usado como uma restrição genérica para os métodos de interface.
// Covariant delegate.
public delegate R DCovariant<out R>();
Um delegado genérico, um tipo pode ser declarado covariante se for usado apenas como um tipo de retorno do
método e não usado para argumentos de método.
Especificação da linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a fonte
definitiva para a sintaxe e o uso de C#.
Consulte também
Variação em Interfaces Genéricas
in
Modificadores
override (Referência de C#)
04/11/2019 • 4 minutes to read • Edit Online
Exemplo
Neste exemplo, a classe Square deve fornecer uma implementação substituída de GetArea porque
GetArea é herdado da classe abstrata Shape :
Um método override fornece uma nova implementação de um membro herdado de uma classe base.
O método que é substituído por uma declaração override é conhecido como o método base
substituído. O método base substituído deve ter a mesma assinatura que o método override . Para
obter informações sobre herança, consulte Herança.
Você não pode substituir um método não virtual ou estático. O método base substituído deve ser
virtual , abstract ou override .
Uma declaração override não pode alterar a acessibilidade do método virtual . O método override
e o método virtual devem ter o mesmo modificador de nível de acesso.
Não é possível usar os modificadores new , static ou virtual para modificar um método override .
Uma declaração de propriedade de substituição deve especificar exatamente o mesmo modificador de
acesso, tipo e nome que a propriedade herdada e a propriedade substituída deve ser virtual ,
abstract ou override .
Para obter mais informações sobre como usar a palavra-chave override , consulte Controle de versão
com as palavras-chave override e new e Quando usar as palavras-chave override e new.
Exemplo
Este exemplo define uma classe base chamada Employee e uma classe derivada chamada
SalesEmployee . A classe SalesEmployee inclui um campo extra, salesbonus , e substitui o método
CalculatePay para levar isso em consideração.
class TestOverride
{
public class Employee
{
public string name;
Consulte também
Referência de C#
Guia de Programação em C#
Herança
Palavras-chave do C#
Modificadores
abstract
virtual
new (modificador)
Polimorfismo
readonly (Referência de C#)
04/11/2019 • 11 minutes to read • Edit Online
WARNING
Um tipo visível externamente que contém um campo somente leitura visível externamente, que é um tipo de
referência mutável, pode ser uma vulnerabilidade de segurança e pode disparar o aviso CA2104 : "Não declare
tipos de referência mutáveis somente leitura".
Você pode atribuir um valor a um campo readonly apenas nos seguintes contextos:
Quando a variável é inicializada na declaração, por exemplo:
NOTE
A palavra-chave readonly é diferente da palavra-chave const. O campo const pode ser inicializado apenas na
declaração do campo. Um campo readonly pode ser atribuído várias vezes na declaração do campo e em qualquer
construtor. Portanto, campos readonly podem ter valores diferentes dependendo do construtor usado. Além disso,
enquanto um campo const é uma constante em tempo de compilação, o campo readonly pode ser usado para
constantes de runtime, como no exemplo a seguir:
public SampleClass()
{
// Initialize a readonly instance field
z = 24;
}
O exemplo anterior usa as propriedades automáticas de readonly para declarar seu armazenamento. Isso
instrui o compilador a criar campos de suporte readonly para essas propriedades. Você também pode declarar
campos readonly diretamente:
Adicionar um campo não marcado como readonly gera o erro do compilador CS8340 : "Os campos da
instância de structs readonly devem ser readonly."
Você tem duas vantagens aplicando o modificador de readonly aos métodos struct aplicáveis. O mais
importante é que o compilador impõe sua intenção. O código que modifica o estado não é válido em um
método readonly . O compilador também pode fazer uso do modificador de readonly para habilitar
otimizações de desempenho. Quando grandes struct tipos são passados por in referência, o compilador
deve gerar uma cópia defensiva se o estado da estrutura puder ser modificado. Se apenas readonly Membros
forem acessados, o compilador poderá não criar a cópia defensiva.
O modificador de readonly é válido na maioria dos membros de um struct , incluindo métodos que
substituem os métodos declarados em System.Object. Há algumas restrições:
Você não pode declarar readonly membros estáticos.
Você não pode declarar construtores de readonly .
Você pode adicionar o modificador de readonly a uma propriedade ou declaração de indexador:
Você também pode adicionar o modificador de readonly a get individuais ou set acessadores de uma
propriedade ou um indexador:
Você não pode adicionar o modificador readonly a uma propriedade e um ou mais dos acessadores da mesma
propriedade. Essa mesma restrição se aplica a indexadores.
O compilador aplica implicitamente o modificador de readonly a propriedades implementadas
automaticamente em que o código implementado pelo compilador não modifica o estado. É equivalente às
seguintes declarações:
Você pode adicionar o modificador de readonly nesses locais, mas ele não terá nenhum efeito significativo.
Você não pode adicionar o modificador readonly a um setter de propriedade autoimplementado ou a uma
propriedade autoimplementada de leitura/gravação.
O tipo retornado não precisa ser um readonly struct . Qualquer tipo que possa ser retornado por ref pode
ser retornado por ref readonly .
Especificação da linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a fonte
definitiva para a sintaxe e o uso de C#.
Você também pode ver as propostas de especificação de idioma:
struct de ref e ReadOnly ReadOnly
Membros de struct ReadOnly
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Modificadores
const
Campos
sealed (Referência de C#)
04/11/2019 • 4 minutes to read • Edit Online
Quando aplicado a uma classe, o modificador sealed impede que outras classes herdem dela. No
exemplo a seguir, a classe B herda da classe A , mas nenhuma classe pode herdar da classe B .
class A {}
sealed class B : A {}
Você também pode usar o modificador sealed em um método ou propriedade que substitui um
método ou propriedade virtual em uma classe base. Com isso, você pode permitir que classes sejam
derivadas de sua classe e impedir que substituam métodos ou propriedades virtuais.
Exemplo
No exemplo a seguir, Z herda de Y , mas Z não pode substituir a função virtual F declarada em X e
lacrada em Y .
class X
{
protected virtual void F() { Console.WriteLine("X.F"); }
protected virtual void F2() { Console.WriteLine("X.F2"); }
}
class Y : X
{
sealed protected override void F() { Console.WriteLine("Y.F"); }
protected override void F2() { Console.WriteLine("Y.F2"); }
}
class Z : Y
{
// Attempting to override F causes compiler error CS0239.
// protected override void F() { Console.WriteLine("Z.F"); }
// Overriding F2 is allowed.
protected override void F2() { Console.WriteLine("Z.F2"); }
}
Quando define novos métodos ou propriedades em uma classe, você pode impedir que classes
derivadas as substituam ao não declará-las como virtuais.
É um erro usar o modificador abstract com uma classe selada, porque uma classe abstrata deve ser
herdada por uma classe que fornece uma implementação dos métodos ou propriedades abstratas.
Quando aplicado a um método ou propriedade, o modificador sealed sempre deve ser usado com
override.
Como structs são lacrados implicitamente, eles não podem ser herdados.
Para obter mais informações, consulte Herança.
Para obter mais exemplos, consulte Classes e membros de classes abstratas e lacradas.
Exemplo
sealed class SealedClass
{
public int x;
public int y;
}
class SealedTest2
{
static void Main()
{
var sc = new SealedClass();
sc.x = 110;
sc.y = 150;
Console.WriteLine($"x = {sc.x}, y = {sc.y}");
}
}
// Output: x = 110, y = 150
No exemplo anterior, você pode tentar herdar da classe lacrada usando a instrução a seguir:
class MyDerivedC: SealedClass {} // Error
Comentários
Para determinar se deve lacrar uma classe, método ou propriedade, geralmente você deve considerar os
dois pontos a seguir:
Os possíveis benefícios que as classes derivadas podem obter por meio da capacidade de
personalizar a sua classe.
O potencial de as classes derivadas modificarem suas classes de forma que elas deixem de
funcionar corretamente ou como esperado.
Especificação da linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é
a fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Classes static e membros de classes static
Classes e membros de classes abstract e sealed
Modificadores de acesso
Modificadores
override
virtual
static (Referência de C#)
04/11/2019 • 5 minutes to read • Edit Online
Use o modificador static para declarar um membro estático que pertença ao próprio tipo, em vez de
um objeto específico. O modificador static pode ser usado com classes, campos, métodos,
propriedades, operadores, eventos e construtores, mas não pode ser usado com indexadores,
finalizadores ou tipos diferentes de classes. Para obter mais informações, consulte Classes Estáticas e
Membros de Classes Estáticas.
Exemplo
A seguinte classe é declarada como static e contém apenas métodos static :
Para fazer referência ao membro estático x , use o nome totalmente qualificado MyBaseC.MyStruct.x ,a
menos que o membro esteja acessível no mesmo escopo:
Console.WriteLine(MyBaseC.MyStruct.x);
Embora uma instância de uma classe contenha uma cópia separada de todos os campos de instância
da classe, há apenas uma cópia de cada campo estático.
Não é possível usar this para referenciar métodos estáticos ou acessadores de propriedade.
Se a palavra-chave static for aplicada a uma classe, todos os membros da classe deverão ser
estáticos.
As classes e as classes estáticas podem ter construtores estáticos. Os construtores estáticos são
chamados em algum ponto entre o momento em que o programa é iniciado e a classe é instanciada.
NOTE
A palavra-chave static tem utilizações mais limitadas do que no C++. Para comparar com a palavra-chave
do C++, consulte Storage classes (C++) (Classes de armazenamento (C++)).
Para demonstrar os membros estáticos, considere uma classe que representa um funcionário da
empresa. Suponha que a classe contém um método para contar funcionários e um campo para
armazenar o número de funcionários. O método e o campo não pertencem a qualquer instância
funcionário. Em vez disso, eles pertencem à classe empresa. Portanto, eles devem ser declarados como
membros estáticos da classe.
Exemplo
Este exemplo lê o nome e a ID de um novo funcionário, incrementa o contador de funcionário em um e
exibe as informações do novo funcionário e do novo número de funcionários. Para simplificar, este
programa lê, do teclado, o número atual de funcionários. Em um aplicativo real, essas informações
devem ser lidas de um arquivo.
public class Employee4
{
public string id;
public string name;
public Employee4()
{
}
Exemplo
Este exemplo mostra que, embora seja possível inicializar um campo estático usando outro campo
estático que ainda não foi declarado, os resultados serão indefinidos até que você atribua
explicitamente um valor ao campo estático.
class Test
{
static int x = y;
static int y = 5;
Test.x = 99;
Console.WriteLine(Test.x);
}
}
/*
Output:
0
5
99
*/
Especificação da linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem
é a fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Modificadores
Classes static e membros de classes static
unsafe (Referência de C#)
04/11/2019 • 2 minutes to read • Edit Online
A palavra-chave unsafe denota um contexto inseguro, que é necessário para qualquer operação que
envolva ponteiros. Para obter mais informações, consulte Código não seguro e ponteiros.
Você pode usar o modificador unsafe na declaração de um tipo ou membro. Toda a extensão textual do
tipo ou membro é, portanto, considerada um contexto inseguro. Por exemplo, a seguir está um método
declarado com o modificador unsafe :
O escopo do contexto inseguro se estende da lista de parâmetros até o final do método, portanto, os
ponteiros também podem ser usados na lista de parâmetros:
unsafe static void FastCopy ( byte* ps, byte* pd, int count ) {...}
Você também pode usar um bloco não seguro para habilitar o uso de um código não seguro dentro desse
bloco. Por exemplo:
unsafe
{
// Unsafe context: can use pointers here.
}
Para compilar o código não seguro, você deve especificar a opção do compilador -unsafe . O código não
seguro não é verificável pelo Common Language Runtime.
Exemplo
// compile with: -unsafe
class UnsafeTest
{
// Unsafe method: takes pointer to int.
unsafe static void SquarePtrParam(int* p)
{
*p *= *p;
}
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Instrução fixed
Código não seguro e ponteiros
Buffers de tamanho fixo
virtual (Referência de C#)
23/10/2019 • 5 minutes to read • Edit Online
A palavra-chave virtual é usada para modificar uma declaração de método, propriedade, indexador ou
evento e permitir que ela seja substituída em uma classe derivada. Por exemplo, esse método pode ser
substituído por qualquer classe que o herde:
A implementação de um membro virtual pode ser alterada por um membro de substituição em uma
classe derivada. Para obter mais informações sobre como usar a palavra-chave virtual , consulte
Controle de versão com as palavras-chave override e new e Quando usar as palavras-chave override e
new.
Comentários
Quando um método virtual é invocado, o tipo de tempo de execução do objeto é verificado para um
membro de substituição. O membro de substituição na classe mais derivada é chamado, que pode ser o
membro original, se nenhuma classe derivada tiver substituído o membro.
Por padrão, os métodos não são virtuais. Você não pode substituir um método não virtual.
Não é possível usar o modificador virtual com os modificadores static , abstract , private ou
override . O exemplo a seguir mostra uma propriedade virtual:
class MyBaseClass
{
// virtual auto-implemented property. Overrides can only
// provide specialized behavior if they implement get and set accessors.
public virtual string Name { get; set; }
As propriedades virtuais se comportam como métodos virtuais, exceto pelas diferenças na declaração e
na sintaxe de invocação.
É um erro usar o modificador virtual em uma propriedade estática.
Uma propriedade herdada virtual pode ser substituída em uma classe derivada incluindo uma
declaração de propriedade que usa o modificador override .
Exemplo
Neste exemplo, a classe Shape contém as duas coordenadas x , y e o método virtual Area() . Classes
de forma diferentes como Circle , Cylinder e Sphere herdam a classe Shape e a área de superfície é
calculada para cada figura. Cada classe derivada tem a própria implementação de substituição do
Area() .
Observe que as classes herdadas Circle , Sphere e Cylinder usam construtores que inicializam a
classe base, conforme mostrado na declaração a seguir.
class TestClass
{
public class Shape
{
public const double PI = Math.PI;
protected double x, y;
public Shape()
{
}
Especificação da linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é
a fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Polimorfismo
abstract
override
new (modificador)
volatile (Referência de C#)
04/11/2019 • 5 minutes to read • Edit Online
A palavra-chave volatile indica que um campo pode ser modificado por vários threads que estão em execução
ao mesmo tempo. O compilador, o sistema do tempo de execução e até mesmo o hardware podem reorganizar as
leituras e gravações para locais de memória por motivos de desempenho. Os campos que são declarados
volatile não estão sujeitos a essas otimizações. A adição do modificador volatile garante que todos os threads
observarão gravações voláteis executadas por qualquer outro thread na ordem em que elas foram executadas.
Não há nenhuma garantia de uma única ordenação total de gravações voláteis como visto em todos os threads de
execução.
A palavra-chave volatile pode ser aplicada a campos desses tipos:
Tipos de referência.
Tipos de ponteiro (em um contexto sem segurança). Observe que embora o ponteiro em si possa ser volátil, o
objeto para o qual ele aponta não pode. Em outras palavras, você não pode declarar um "ponteiro como volátil".
Tipos simples, como sbyte , byte , short , ushort , int , uint , char , float e bool .
Um tipo enum com um dos seguintes tipos base: byte , sbyte , short , ushort , int ou uint .
Parâmetros de tipo genérico conhecidos por serem tipos de referência.
IntPtr e UIntPtr.
Outros tipos, inclusive double e long , não podem ser marcados como volatile , pois as leituras e gravações nos
campos desses tipos não podem ser garantidas como atômicas. Para proteger o acesso multithreaded a esses tipos
de campo, use os membros da classe Interlocked ou proteja o acesso usando a instrução lock .
A palavra-chave volatile pode ser aplicada somente aos campos de uma class ou struct . As variáveis locais
não podem ser declaradas como volatile .
Exemplo
O exemplo a seguir mostra como declarar uma variável de campo público como volatile .
class VolatileTest
{
public volatile int sharedStorage;
O exemplo a seguir demonstra como um thread de trabalho ou auxiliar pode ser criado e usado para executar o
processamento em paralelo com o do thread primário. Para saber mais sobre multithreading, confira Threading
gerenciado.
public class Worker
{
// This method is called when the thread is started.
public void DoWork()
{
bool work = false;
while (!_shouldStop)
{
work = !work; // simulate some work
}
Console.WriteLine("Worker thread: terminating gracefully.");
}
public void RequestStop()
{
_shouldStop = true;
}
// Keyword volatile is used as a hint to the compiler that this data
// member is accessed by multiple threads.
private volatile bool _shouldStop;
}
Com o modificador volatile adicionado à declaração de _shouldStop definida, você sempre obterá os mesmos
resultados (semelhante ao trecho mostrado no código anterior). No entanto, sem esse modificador no membro
_shouldStop , o comportamento é imprevisível. O método DoWork pode otimizar o acesso do membro, resultando
na leitura de dados obsoletos. Devido à natureza da programação multithreaded, o número de leituras obsoletas é
imprevisível. Diferentes execuções do programa produzirão resultados um pouco diferentes.
Especificação da linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a fonte
definitiva para a sintaxe e o uso de C#.
Consulte também
Especificação da linguagem C#: palavra-chave volatile
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Modificadores
Instrução lock
Interlocked
Palavras-chave de instrução (Referência de C#)
23/10/2019 • 2 minutes to read • Edit Online
As instruções são instruções do programa. Exceto conforme descrito nos tópicos referenciados na tabela a seguir,
as instruções são executadas em sequência. A tabela a seguir lista as palavras-chave da instrução C#. Para obter
mais informações sobre instruções que não são expressos com qualquer palavra-chave, consulte Instruções.
CATEGORIA PALAVRAS-CHAVE DE C#
Consulte também
Referência de C#
Instruções
Palavras-chave do C#
if-else (Referência de C#)
23/10/2019 • 8 minutes to read • Edit Online
Uma instrução if identifica qual instrução executar com base no valor de uma expressão booliana. No
exemplo a seguir, a variável bool condition deve ser definida como true e, em seguida, verificada na
instrução if . A saída é The variable is set to true. .
if (condition)
{
Console.WriteLine("The variable is set to true.");
}
else
{
Console.WriteLine("The variable is set to false.");
}
Você pode executar os exemplos neste tópico colocando-os no método Main de um aplicativo de console.
Uma instrução if no C# pode ocorrer de duas formas, como mostra o exemplo a seguir.
// if-else statement
if (condition)
{
then-statement;
}
else
{
else-statement;
}
// Next statement in the program.
Em uma instrução if-else , se condition for avaliada como verdadeira, a then-statement será executada. Se
condition for falsa, a else-statement será executada. Como condition não pode ser verdadeira e falsa
simultaneamente, a then-statement e a else-statement de uma instrução if-else nunca podem ser
executadas juntas. Após then-statement ou else-statement ser executada, o controle é transferido para a
próxima instrução após a instrução if .
Em uma instrução if que não inclui uma instrução else , se condition for verdadeira, a then-statement será
executada. Se condition for falsa, o controle será transferido para a próxima instrução após a instrução if .
Tanto a then-statement como a else-statement podem consistir em uma única instrução ou várias instruções
entre chaves ( {} ). Para uma única instrução, as chaves são opcionais, mas recomendadas.
A instrução ou instruções na then-statement e na else-statement podem ser de qualquer tipo, incluindo outra
instrução if aninhada na instrução if original. Em instruções if aninhadas, cada cláusula else pertence
ao último if que não tem um else correspondente. No exemplo a seguir, Result1 aparecerá se m > 10 e
n > 20 forem avaliadas como verdadeiras. Se m > 10 for verdadeiro, mas n > 20 for falso, Result2 será
exibido.
if (m > 10)
if (n > 20)
{
Console.WriteLine("Result1");
}
else
{
Console.WriteLine("Result2");
}
Se, em vez disso, você desejar que Result2 apareça quando (m > 10) for falso, pode especificar essa
associação usando chaves para estabelecer o início e o fim da instrução if aninhada, como o exemplo a
seguir mostra.
Exemplo
No exemplo a seguir, você insere um caractere do teclado e o programa usa uma instrução if aninhada para
determinar se o caractere de entrada é um caractere alfabético. Se o caractere de entrada for um caractere
alfabético, o programa verificará se o caractere de entrada está em maiúscula ou minúscula. Será exibida uma
mensagem para cada caso.
Console.Write("Enter a character: ");
char c = (char)Console.Read();
if (Char.IsLetter(c))
{
if (Char.IsLower(c))
{
Console.WriteLine("The character is lowercase.");
}
else
{
Console.WriteLine("The character is uppercase.");
}
}
else
{
Console.WriteLine("The character isn't an alphabetic character.");
}
//Sample Output:
//Enter a character: 2
//The character isn't an alphabetic character.
//Enter a character: A
//The character is uppercase.
//Enter a character: h
//The character is lowercase.
Exemplo
Você também pode aninhar uma instrução if dentro de um bloco else, como mostra o código parcial a seguir.
O exemplo aninha instruções if em dois blocos else e um bloco then. Os comentários especificam quais
condições são verdadeiras ou falsas em cada bloco.
// Change the values of these variables to test the results.
bool Condition1 = true;
bool Condition2 = true;
bool Condition3 = true;
bool Condition4 = true;
if (Condition1)
{
// Condition1 is true.
}
else if (Condition2)
{
// Condition1 is false and Condition2 is true.
}
else if (Condition3)
{
if (Condition4)
{
// Condition1 and Condition2 are false. Condition3 and Condition4 are true.
}
else
{
// Condition1, Condition2, and Condition4 are false. Condition3 is true.
}
}
else
{
// Condition1, Condition2, and Condition3 are false.
}
Exemplo
O exemplo a seguir determina se um caractere de entrada é um número, uma letra maiúscula ou uma letra
minúscula. Se todas as três condições forem falsas, o caractere não será um caractere alfanumérico. O exemplo
exibe uma mensagem para cada caso.
Console.Write("Enter a character: ");
char ch = (char)Console.Read();
if (Char.IsUpper(ch))
{
Console.WriteLine("The character is an uppercase letter.");
}
else if (Char.IsLower(ch))
{
Console.WriteLine("The character is a lowercase letter.");
}
else if (Char.IsDigit(ch))
{
Console.WriteLine("The character is a number.");
}
else
{
Console.WriteLine("The character is not alphanumeric.");
}
//Enter a character: e
//The character is a lowercase letter.
//Enter a character: 4
//The character is a number.
//Enter a character: =
//The character is not alphanumeric.
Assim como uma instrução no bloco else ou no bloco then pode ser qualquer instrução válida, você pode usar
qualquer expressão booliana válida para a condição. Você pode usar operadores lógicos tais como ! , && , ||
, & , | , e ^ para criar condições compostas. O código a seguir mostra exemplos.
// NOT
bool result = true;
if (!result)
{
Console.WriteLine("The condition is true (result is false).");
}
else
{
Console.WriteLine("The condition is false (result is true).");
}
// Short-circuit AND
int m = 9;
int n = 7;
int p = 5;
if (m >= n && m >= p)
{
Console.WriteLine("Nothing is larger than m.");
}
// Short-circuit OR
if (m > n || m > p)
{
Console.WriteLine("m isn't the smallest.");
}
// NOT and OR
m = 4;
if (!(m >= n || m >= p))
{
Console.WriteLine("Now m is the smallest.");
}
// Output:
// The condition is false (result is true).
// Nothing is larger than m.
// Nothing is larger than m.
// m isn't the smallest.
// Now m is the smallest.
Especificação da linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a
fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
?: ??
Instrução if-else (C++)
switch
switch (Referência em C#)
29/11/2019 • 25 minutes to read • Edit Online
switch é uma instrução de seleção que escolhe uma única seção switch para ser executada de uma lista
de candidatas com base em uma correspondência de padrão com a expressão de correspondência.
using System;
switch (caseSwitch)
{
case 1:
Console.WriteLine("Case 1");
break;
case 2:
Console.WriteLine("Case 2");
break;
default:
Console.WriteLine("Default case");
break;
}
}
}
// The example displays the following output:
// Case 1
A instrução switch geralmente é usada como uma alternativa para um constructo if-else se uma única
expressão é testada com três ou mais condições. Por exemplo, a instrução switch a seguir determina se
uma variável do tipo Color tem um dos três valores:
using System;
using System;
A expressão de correspondência
A expressão de correspondência fornece o valor para corresponder aos padrões nos rótulos case .A
sintaxe é:
switch (expr)
No C# 6 e versões anteriores, a expressão de correspondência deve ser uma expressão que retorna um
valor dos seguintes tipos:
um char.
um string.
um bool.
um valor integral , como um int ou um long .
um valor enum.
Começando com o C# 7.0, a expressão de correspondência pode ser qualquer expressão não nula.
A seção switch
Uma instrução switch inclui uma ou mais seções do comutador. Cada seção switch contém um ou mais
rótulos case (em um rótulo case ou padrão) seguidos por uma ou mais instruções. A instrução switch
pode incluir no máximo um rótulo padrão colocado em qualquer seção switch. O exemplo a seguir
mostra uma instrução switch simples que tem três seções switch, cada uma contendo duas instruções.
A segunda seção switch contém os rótulos case 2: e case 3: .
Uma instrução switch pode incluir qualquer número de seções switch e cada seção pode ter um ou
mais rótulos case, conforme mostrado no exemplo a seguir. No entanto, dois rótulos case não podem
conter a mesma expressão.
using System;
switch (caseSwitch)
{
case 1:
Console.WriteLine("Case 1");
break;
case 2:
case 3:
Console.WriteLine($"Case {caseSwitch}");
break;
default:
Console.WriteLine($"An unexpected value ({caseSwitch})");
break;
}
}
}
// The example displays output like the following:
// Case 1
Apenas uma seção switch em uma instrução switch é executada. O C# não permite que a execução
continue de uma seção switch para a próxima. Por isso, o código a seguir gera um erro do compilador,
CS0163: "o controle não pode passar de um rótulo case (<rótulo case >) para outro".
switch (caseSwitch)
{
// The following switch section causes an error.
case 1:
Console.WriteLine("Case 1...");
// Add a break or other jump statement here.
case 2:
Console.WriteLine("... and/or Case 2");
break;
}
Esse requisito é atendido normalmente saindo explicitamente da seção switch usando uma instrução
break, goto ou return. No entanto, o código a seguir também é válido porque garante que o controle do
programa não pode passar para a seção switch default .
switch (caseSwitch)
{
case 1:
Console.WriteLine("Case 1...");
break;
case 2:
case 3:
Console.WriteLine("... and/or Case 2");
break;
case 4:
while (true)
Console.WriteLine("Endless looping. . . .");
default:
Console.WriteLine("Default value...");
break;
}
A execução da lista de instruções na seção switch com um rótulo case que corresponde à expressão de
correspondência começa com a primeira instrução e continua pela lista de instruções, normalmente até
uma instrução de salto como break , goto case , goto label , return ou throw ser atingida. Nesse
ponto, o controle é transferido para fora da instrução switch ou para outro rótulo case. Se uma
instrução goto for usada, ela deverá transferir controle para um rótulo de constante. Essa restrição é
necessária, já que tentar transferir o controle para um rótulo não constante pode ter efeitos colaterais
indesejáveis, tais como transferência do controle para um local não intencional no código ou criação de
um loop infinito.
Rótulos case
Cada rótulo case especifica um padrão a ser comparado com a expressão de correspondência (a variável
caseSwitch nos exemplos anteriores). Se eles corresponderem, o controle será transferido para a seção
switch que contém o primeiro rótulo case correspondente. Se nenhum padrão de rótulo de caso
corresponder à expressão de correspondência, o controle será transferido para a seção com o rótulo de
caso default , se existir algum. Se não houver nenhum case default , nenhuma declaração em qualquer
seção switch será executada e o controle será transferido para fora da instrução switch .
Para obter informações sobre a instrução switch e a correspondência de padrões, consulte a seção
Correspondência de padrões com a instrução switch .
Como o C# 6 dá suporte apenas ao padrão de constante e não permite a repetição de valores de
constantes, os rótulos case definem valores mutuamente exclusivos e apenas um padrão pode
corresponder à expressão de correspondência. Como resultado, a ordem na qual as instruções case
aparecem não é importante.
No entanto, no C# 7.0, como há suporte para outros padrões, os rótulos case não precisam definir
valores mutuamente exclusivos e vários padrões podem corresponder à expressão de correspondência.
Como são executadas apenas as instruções na primeira seção switch que contém o primeiro padrão de
correspondência, a ordem na qual as instruções case aparecem agora é importante. Se o C# detecta
uma seção switch cujas instruções case são equivalentes ou são subconjuntos de instruções anteriores,
ele gera um erro do compilador, CS8120, “O switch case já foi tratado por um case anterior”.
O exemplo a seguir ilustra uma instrução switch que usa uma variedade de padrões não mutuamente
exclusivos. Se você mover a seção switch case 0: para que ela não seja mais a primeira seção na
instrução switch , o C# gera um erro do compilador porque um inteiro cujo valor é zero é um
subconjunto de todos os inteiros, que é o padrão definido pela instrução case int val .
using System;
using System.Collections.Generic;
using System.Linq;
Você pode corrigir esse problema e eliminar o aviso do compilador em uma das duas maneiras:
Alterando a ordem das seções switch.
Usando uma cláusula where no rótulo case .
O case default
O case default especifica a seção switch a ser executada se a expressão de correspondência não
corresponder a nenhum outro rótulo case . Se um case default não estiver presente e a expressão de
correspondência não corresponder a nenhum outro rótulo case , o fluxo do programa passará pela
instrução switch .
O case default pode aparecer em qualquer ordem na instrução switch . Independentemente de sua
ordem no código-fonte, ele é sempre avaliado por último, afinal os rótulos case foram avaliados.
case constant:
em que constant é o valor para testar. constant pode ser qualquer uma das expressões de constante a
seguir:
Um literal bool : seja true ou false .
Qualquer constante integral , como uma int , uma long ou uma byte .
O nome de uma variável const declarada.
Uma constante de enumeração.
Um literal char.
Um literal string.
A expressão de constante é avaliada da seguinte forma:
Se expr e constant forem tipos integrais, o operador de igualdade de C# determinará se a
expressão retorna true (ou seja, se expr == constant ).
Caso contrário, o valor da expressão será determinado por uma chamada ao método estático
Object.Equals(expr, constant).
O exemplo a seguir usa o padrão de constante para determinar se uma data específica é um final de
semana, o primeiro dia, o último dia ou o meio da semana de trabalho. Ele avalia a propriedade
DateTime.DayOfWeek do dia atual em relação aos membros da enumeração DayOfWeek.
using System;
class Program
{
static void Main()
{
switch (DateTime.Now.DayOfWeek)
{
case DayOfWeek.Sunday:
case DayOfWeek.Saturday:
Console.WriteLine("The weekend");
break;
case DayOfWeek.Monday:
Console.WriteLine("The first day of the work week.");
break;
case DayOfWeek.Friday:
Console.WriteLine("The last day of the work week.");
break;
default:
Console.WriteLine("The middle of the work week.");
break;
}
}
}
// The example displays output like the following:
// The middle of the work week.
O exemplo a seguir usa o padrão de constante para manipular a entrada do usuário em um aplicativo de
console que simula uma máquina de café automática.
using System;
class Example
{
static void Main()
{
Console.WriteLine("Coffee sizes: 1=small 2=medium 3=large");
Console.Write("Please enter your selection: ");
string str = Console.ReadLine();
int cost = 0;
Padrão de tipo
O padrão de tipo permite a conversão e a avaliação do tipo concisa. Quando usado com a instrução
switch para realizar a correspondência de padrões, testa se uma expressão pode ser convertida para
um tipo especificado e, se puder, o converte em uma variável desse tipo. A sintaxe é:
em que type é o nome do tipo no qual o resultado de expr deverá ser convertido e varname é o objeto
para o qual o resultado de expr é convertido se a correspondência é bem-sucedida. O tipo de tempo de
compilação expr pode ser um parâmetro de tipo genérico, começando com C# 7.1.
A expressão case será true se qualquer uma das condições seguintes for verdadeira:
expr for uma instância do mesmo tipo que type.
expr for uma instância de um tipo derivado de type. Em outras palavras, o resultado de expr pode
sofrer upcast para uma instância de type.
expr tem um tipo de tempo de compilação que é uma classe base de type, e expr tem um tipo de
runtime que é type ou é derivado de type. O tipo do tempo de compilação de uma variável é o
tipo da variável conforme definido em sua declaração de tipo. O tipo de runtime de uma variável
é o tipo da instância atribuída a essa variável.
expr é uma instância de um tipo que implementa a interface de type.
Se a expressão case for true, varname será definitivamente atribuído e terá o escopo local dentro da
seção switch apenas.
Observe que null não corresponde a um tipo. Para corresponder um null , use o seguinte rótulo
case :
case null:
O exemplo a seguir usa o padrão de tipo para fornecer informações sobre vários tipos de coleção.
using System;
using System.Collections;
using System.Collections.Generic;
using System.Linq;
class Example
{
static void Main(string[] args)
{
int[] values = { 2, 4, 6, 8, 10 };
ShowCollectionInformation(values);
Em vez de object , você poderia empregar um método genérico, usando o tipo da coleção como o
parâmetro de tipo, conforme mostrado no código a seguir:
using System;
using System.Collections;
using System.Collections.Generic;
using System.Linq;
class Example
{
static void Main(string[] args)
{
int[] values = { 2, 4, 6, 8, 10 };
ShowCollectionInformation(values);
A versão genérica é diferente da primeira amostra de duas maneiras. Primeiro, você não pode usar o
case null . Você não pode usar nenhum case constante porque o compilador não pode converter
qualquer tipo arbitrário de T para qualquer tipo diferente de object . O que tinha sido o case default
agora testa para um object não nulo. Isso significa que o case default testa apenas para null .
Sem a correspondência de padrões, esse código pode ser escrito da seguinte maneira. O uso da
correspondência de padrões de tipo produz um código mais compacto e legível eliminando a
necessidade de testar se o resultado de uma conversão é um null ou executar conversões repetidas.
using System;
using System.Collections;
using System.Collections.Generic;
using System.Linq;
class Example
{
static void Main(string[] args)
{
int[] values = { 2, 4, 6, 8, 10 };
ShowCollectionInformation(values);
using System;
Observe que a cláusula when no exemplo que tenta testar se um objeto Shape é null não é executada.
O padrão de tipo correto para testar um null é case null: .
Especificação da linguagem C#
Para obter mais informações, consulte A instrução switch na Especificação da linguagem C#. A
especificação da linguagem é a fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
if-else
Correspondência Padrão
do (Referência de C#)
29/11/2019 • 2 minutes to read • Edit Online
A instrução do executa uma instrução ou um bloco de instruções enquanto uma expressão booliana
especificada é avaliada como true . Como essa expressão é avaliada após cada execução do loop, um loop
do-while é executado uma ou mais vezes. Isso é diferente do loop while, que é executado zero ou mais vezes.
A qualquer momento dentro do bloco de instruções do , interrompa o loop usando a instrução break.
Você pode seguir diretamente para a avaliação da expressão while usando a instrução continue. Se a
expressão for avaliada como true , a execução continuará na primeira instrução do loop. Caso contrário, a
execução continuará na primeira instrução após o loop.
Você também pode sair de um loop do-while com a instrução goto, return ou throw.
Exemplo
O exemplo a seguir mostra o uso da instrução do . Selecione Executar para executar o código de exemplo.
Depois disso, você pode modificar o código e executá-lo novamente.
int n = 0;
do
{
Console.WriteLine(n);
n++;
} while (n < 5);
Especificação da linguagem C#
Para saber mais, confira a seção A instrução do na Especificação da linguagem C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
instrução while
for (referência de C#)
29/11/2019 • 5 minutes to read • Edit Online
A instrução for executa uma instrução ou um bloco de instruções enquanto uma expressão booliana
especificada é avaliada como true .
Em qualquer ponto dentro do bloco de instrução for , você pode sair do loop usando a instrução break ou
seguir para a próxima iteração no loop usando a instrução continue. Você também pode sair de um loop
for com a instrução goto, return ou throw.
Todas as três seções são opcionais. O corpo do loop é uma instrução ou um bloco de instruções.
A exemplo a seguir mostra a instrução for com todas as seções definidas:
A seção inicializador
As instruções na seção de inicializador são executadas apenas uma vez, antes de entrar no loop. A seção
inicializador é uma das seguintes:
A declaração e a inicialização de uma variável de loop local, que não pode ser acessada de fora do
loop.
Zero ou mais expressões de instrução da lista a seguir, separadas por vírgulas:
instrução de atribuição
invocação de um método
prefixo ou sufixo da expressão incrementar, como ++i ou i++
int i = 0
A seção condição
A seção condição, se presente, deverá ser uma expressão booliana. Essa expressão é avaliada antes de cada
iteração do loop. Se a seção condição não estiver presente ou a expressão booliana for avaliada como true ,
a próxima iteração do loop será executada; caso contrário, o loop será finalizado.
A seção condição no exemplo acima determina se o loop será encerrado com base no valor da variável de
loop local:
i < 5
A seção iterador
A seção iterador define o que acontece depois de cada iteração do corpo do loop. A seção iterador contém
zero ou mais das seguintes expressões de instrução, separadas por vírgulas:
instrução de atribuição
invocação de um método
prefixo ou sufixo da expressão incrementar, como ++i ou i++
i++
Exemplos
O exemplo a seguir ilustra vários usos menos comuns das seções de instrução for : atribuir um valor a uma
variável de loop externa na seção inicializador, invocar um método inicializa nas seções de inicializador e de
iterador e alterar os valores de duas variáveis na seção de iterador. Selecione Executar para executar o
código de exemplo. Depois disso, você pode modificar o código e executá-lo novamente.
int i;
int j = 10;
for (i = 0, Console.WriteLine($"Start: i={i}, j={j}"); i < j; i++, j--, Console.WriteLine($"Step: i={i},
j={j}"))
{
// Body of the loop.
}
for ( ; ; )
{
// Body of the loop.
}
Especificação da linguagem C#
Para obter mais informações, confira a seção A instrução for na Especificação da linguagem C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
foreach, in
foreach, in (Referência em C#)
04/11/2019 • 4 minutes to read • Edit Online
A instrução foreach executa uma instrução ou um bloco de instruções para cada elemento em uma
instância do tipo que implementa a interface System.Collections.IEnumerable ou
System.Collections.Generic.IEnumerable<T>. A instrução foreach não está limitada a esses tipos e
pode ser aplicada a uma instância de qualquer tipo que satisfaça as seguintes condições:
incluir um método GetEnumerator público sem parâmetros, cujo tipo de retorno é um tipo de classe,
estrutura ou interface,
o tipo de retorno do método GetEnumerator tem a propriedade Current pública e o método
MoveNext público sem parâmetros, cujo tipo de retorno é Boolean.
Exemplos
NOTE
Os exemplos de C# neste artigo são executados no executador de código embutido Try.NET e no playground.
Clique no botão Executar para executar um exemplo em uma janela interativa. Ao executar o código, é possível
modificá-lo e executar o código modificado clicando em Executar novamente. O código modificado será
executado na janela interativa ou, se a compilação falhar, a janela interativa exibirá todos as mensagens de erro
do compilador C#.
O exemplo a seguir mostra o uso da instrução foreach com uma instância do tipo List<T> que
implementa a interface IEnumerable<T>:
O exemplo a seguir usa a instrução foreach com uma instância do tipo System.Span<T>, que não
implementa nenhuma interface:
public class IterateSpanExample
{
public static void Main()
{
Span<int> numbers = new int[] { 3, 14, 15, 92, 6 };
foreach (int number in numbers)
{
Console.Write($"{number} ");
}
Console.WriteLine();
}
}
O exemplo a seguir usa uma variável de iteração ref para definir o valor de cada item em uma matriz
stackalloc. A versão ref readonly itera a coleção para imprimir todos os valores. A declaração
readonly usa uma declaração de variável local implícita. Declarações de variável implícita podem ser
usadas com declarações ref ou ref readonly , assim como declarações de variável tipadas
explicitamente.
Especificação da linguagem C#
Para obter mais informações, confira a seção A instrução foreach na Especificação da linguagem C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Usar foreach com matrizes
instrução for
while (Referência de C#)
29/11/2019 • 2 minutes to read • Edit Online
A instrução while executa uma instrução ou um bloco de instruções enquanto uma expressão booliana
especificada é avaliada como true . Como essa expressão é avaliada antes de cada execução do loop, um loop
while é executado zero ou mais vezes. Isso difere do loop do, que é executado uma ou mais vezes.
A qualquer momento dentro do bloco de instruções while , interrompa o loop usando a instrução break.
Você pode seguir diretamente para a avaliação da expressão while usando a instrução continue. Se a
expressão for avaliada como true , a execução continuará na primeira instrução do loop. Caso contrário, a
execução continuará na primeira instrução após o loop.
Você também pode sair de um loop while com a instrução goto, return ou throw.
Exemplo
O exemplo a seguir mostra o uso da instrução while . Selecione Executar para executar o código de exemplo.
Depois disso, você pode modificar o código e executá-lo novamente.
int n = 0;
while (n < 5)
{
Console.WriteLine(n);
n++;
}
Especificação da linguagem C#
Para saber mais, confira a seção A instrução while na Especificação da linguagem C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Instrução do
break (Referência de C#)
23/10/2019 • 3 minutes to read • Edit Online
A instrução break termina o loop delimitador mais próximo ou a instrução switch na qual ele aparece.
Controle é passado para a instrução que segue a instrução encerrada, se houver.
Exemplo
Neste exemplo, a instrução condicional tem um contador que deveria contar de 1 a 100. No entanto, a
instrução break encerra o loop após 4 contagens.
class BreakTest
{
static void Main()
{
for (int i = 1; i <= 100; i++)
{
if (i == 5)
{
break;
}
Console.WriteLine(i);
}
Exemplo
Este exemplo demonstra o uso de break em uma instrução switch.
class Switch
{
static void Main()
{
Console.Write("Enter your selection (1, 2, or 3): ");
string s = Console.ReadLine();
int n = Int32.Parse(s);
switch (n)
{
case 1:
Console.WriteLine("Current value is 1");
break;
case 2:
Console.WriteLine("Current value is 2");
break;
case 3:
Console.WriteLine("Current value is 3");
break;
default:
Console.WriteLine("Sorry, invalid selection.");
break;
}
Sample Output:
Enter your selection (1, 2, or 3): 1
Current value is 1
*/
Exemplo
Neste exemplo, a instrução break é usada para interromper um loop aninhado interno e retornar o controle
para o loop externo. O controle só retornou um nível acima nos loops aninhados.
class BreakInNestedLoops
{
static void Main(string[] args)
{
int[] numbers = { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9 };
char[] letters = { 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j' };
// Outer loop.
for (int i = 0; i < numbers.Length; i++)
{
Console.WriteLine($"num = {numbers[i]}");
// Inner loop.
for (int j = 0; j < letters.Length; j++)
{
if (j == i)
{
// Return control to outer loop.
break;
}
Console.Write($" {letters[j]} ");
}
Console.WriteLine();
}
/*
* Output:
num = 0
num = 1
a
num = 2
a b
num = 3
a b c
num = 4
a b c d
num = 5
a b c d e
num = 6
a b c d e f
num = 7
a b c d e f g
num = 8
a b c d e f g h
num = 9
a b c d e f g h i
*/
Exemplo
Neste exemplo, a instrução break é usada apenas para interromper a ramificação atual durante cada iteração
do loop. O loop em si não é afetado pelas instâncias de break que pertencem à instrução switch aninhada.
class BreakFromSwitchInsideLoop
{
static void Main(string[] args)
{
// loop 1 to 3
for (int i = 1; i <= 3; i++)
{
switch(i)
{
case 1:
Console.WriteLine("Current value is 1");
break;
case 2:
Console.WriteLine("Current value is 2");
break;
case 3:
Console.WriteLine("Current value is 3");
break;
default:
Console.WriteLine("This shouldn't happen.");
break;
}
}
/*
* Output:
Current value is 1
Current value is 2
Current value is 3
*/
Especificação da linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a
fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
switch
continue (Referência de C#)
23/10/2019 • 2 minutes to read • Edit Online
A instrução continue passa o controle para a próxima iteração da instrução de circunscrição while, do, for ou
foreach na qual ela aparece.
Exemplo
Neste exemplo, um contador é inicializado para a contagem de 1 a 10. Ao usar a instrução continue junto
com a expressão (i < 9) , as instruções entre continue e o término do corpo for serão ignoradas.
class ContinueTest
{
static void Main()
{
for (int i = 1; i <= 10; i++)
{
if (i < 9)
{
continue;
}
Console.WriteLine(i);
}
Especificação da linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a
fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Instrução Break
goto (Referência de C#)
23/10/2019 • 2 minutes to read • Edit Online
A declaração goto transfere o controle do programa diretamente para uma instrução rotulada.
Um uso comum de goto é a transferência do controle para um rótulo de caso de opção específico ou para o
rótulo padrão em uma instrução switch .
A instrução goto também é útil para sair de loops profundamente aninhados.
Exemplo
O exemplo a seguir demonstra o uso de goto em uma instrução switch.
class SwitchTest
{
static void Main()
{
Console.WriteLine("Coffee sizes: 1=Small 2=Medium 3=Large");
Console.Write("Please enter your selection: ");
string s = Console.ReadLine();
int n = int.Parse(s);
int cost = 0;
switch (n)
{
case 1:
cost += 25;
break;
case 2:
cost += 25;
goto case 1;
case 3:
cost += 50;
goto case 1;
default:
Console.WriteLine("Invalid selection.");
break;
}
if (cost != 0)
{
Console.WriteLine($"Please insert {cost} cents.");
}
Console.WriteLine("Thank you for your business.");
Sample Output:
Coffee sizes: 1=Small 2=Medium 3=Large
Please enter your selection: 2
Please insert 50 cents.
Thank you for your business.
*/
Exemplo
O exemplo a seguir demonstra o uso de goto para sair de loops aninhados.
// Read input.
Console.Write("Enter the number to search for: ");
// Input a string.
string myNumber = Console.ReadLine();
// Search.
for (int i = 0; i < x; i++)
{
for (int j = 0; j < y; j++)
{
if (array[i, j].Equals(myNumber))
{
goto Found;
}
}
}
Found:
Console.WriteLine($"The number {myNumber} is found.");
Finish:
Console.WriteLine("End of search.");
Sample Output
Enter the number to search for: 44
The number 44 is found.
End of search.
*/
Especificação da linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a
fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Instrução goto (C++)
return (Referência de C#)
23/10/2019 • 2 minutes to read • Edit Online
A instrução return finaliza a execução do método em que aparece e retorna o controle para o método de
chamada. Ela também pode retornar um valor opcional. Se o método for um tipo void , a instrução
return poderá ser omitida.
Se a instrução return estiver dentro de um bloco try , o bloco finally , se houver, será executado antes
que o controle retorne para o método de chamada.
Exemplo
No exemplo a seguir, o método CalculateArea() retorna a variável local area como um valor double .
class ReturnTest
{
static double CalculateArea(int r)
{
double area = r * r * Math.PI;
return area;
}
Especificação da linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a
fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Instrução return
throw (Referência de C#)
31/10/2019 • 5 minutes to read • Edit Online
Comentários
A sintaxe de throw é:
throw [e];
em que e é uma instância de uma classe derivada de System.Exception. O exemplo a seguir usará a
instrução throw para gerar um IndexOutOfRangeException se o argumento passado para um método
chamado GetNumber não corresponder a um índice válido de uma matriz interna.
using System;
Os chamadores do método, então, usam um bloco try-catch ou try-catch-finally para tratar a exceção
gerada. O exemplo a seguir trata a exceção gerada pelo método GetNumber .
using System;
using System;
IMPORTANT
Também é possível usar a sintaxe throw e em um bloco catch para instanciar uma nova exceção que você passa
para o chamador. Nesse caso, o rastreamento de pilha da exceção original, que fica disponível na propriedade
StackTrace, não é preservado.
A expressão throw
Começando com o C# 7.0, o throw pode ser usado como uma expressão, bem como uma instrução. Isso
permite que uma exceção seja gerada em contextos que não tinham suporte anteriormente. Elas incluem:
o operador condicional. O exemplo a seguir usará uma expressão throw para gerar um
ArgumentException se uma matriz de cadeia de caracteres vazia for passada para um método. Antes
do C# 7.0, essa lógica precisava aparecer em uma instrução if / else .
private static void DisplayFirstNumber(string[] args)
{
string arg = args.Length >= 1 ? args[0] :
throw new ArgumentException("You must supply an argument");
if (Int64.TryParse(arg, out var number))
Console.WriteLine($"You entered {number:F0}");
else
Console.WriteLine($"{arg} is not a number.");
o operador de união nula. No exemplo a seguir, uma expressão throw será usada com um operador
de união nula para gerar uma exceção se a cadeia de caracteres atribuída a uma propriedade Name
for null .
um método ou lambda com corpo de expressão. O exemplo a seguir ilustra um método com corpo de
expressão que gera um InvalidCastException porque uma conversão para um valor DateTime não
tem suporte.
Especificação da linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a
fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
try-catch
Palavras-chave do C#
Como gerar exceções explicitamente
try-catch (Referência de C#)
23/10/2019 • 15 minutes to read • Edit Online
A instrução try-catch consiste em um bloco try seguido por uma ou mais cláusulas catch , que
especificam os manipuladores para diferentes exceções.
Quando uma exceção é lançada, o CLR (Common Language Runtime) procura a instrução catch que trata
essa exceção. Se o método em execução no momento não contiver um bloco catch , o CLR procurará no
método que chamou o método atual e assim por diante para cima na pilha de chamadas. Se nenhum bloco
catch for encontrado, o CLR exibirá uma mensagem de exceção sem tratamento para o usuário e
interromperá a execução do programa.
O bloco try contém o código protegido que pode causar a exceção. O bloco é executado até que uma
exceção seja lançada ou ele seja concluído com êxito. Por exemplo, a tentativa a seguir de converter um
objeto null gera a exceção NullReferenceException:
object o2 = null;
try
{
int i2 = (int)o2; // Error
}
Embora a cláusula catch possa ser usada sem argumentos para capturar qualquer tipo de exceção, esse
uso não é recomendado. Em geral, você deve capturar apenas as exceções das quais você sabe se
recuperar. Portanto, você sempre deve especificar um argumento de objeto derivado de System.Exception,
por exemplo:
catch (InvalidCastException e)
{
}
É possível usar mais de uma cláusula catch específica na mesma instrução try-catch. Nesse caso, a ordem
das cláusulas catch é importante porque as cláusulas catch são examinadas em ordem. Capture as
exceções mais específicas antes das menos específicas. O compilador gerará um erro se você ordenar os
blocos catch de forma que um bloco posterior nunca possa ser alcançado.
Usar argumentos catch é uma maneira de filtrar as exceções que deseja manipular. Use também um filtro
de exceção que examina melhor a exceção para decidir se ela deve ser manipulada. Se o filtro de exceção
retornar falso, a pesquisa por um manipulador continuará.
Os filtros de exceção são preferíveis em relação à captura e relançamento (explicados abaixo) porque os
filtros deixam a pilha intacta. Se um manipulador posterior despeja a pilha, você pode ver de onde a
exceção originalmente veio, em vez de apenas o último lugar em que ela foi relançada. Um uso comum de
expressões de filtro de exceção é o registro em log. Crie um filtro que sempre retorna falso e que também
gera um log; você pode registrar exceções conforme elas ocorrem sem precisar manipulá-las e gerá-las
novamente.
Uma instrução throw pode ser usada em um bloco catch para relançar a exceção que foi capturada pela
instrução catch . O exemplo a seguir extrai informações de origem de uma exceção IOException e, em
seguida, lança a exceção para o método pai.
catch (FileNotFoundException e)
{
// FileNotFoundExceptions are handled here.
}
catch (IOException e)
{
// Extract some information from this exception, and then
// throw it to the parent method.
if (e.Source != null)
Console.WriteLine("IOException source: {0}", e.Source);
throw;
}
Você pode capturar uma exceção e lançar uma exceção diferente. Quando fizer isso, especifique a exceção
capturada como a exceção interna, como mostrado no exemplo a seguir.
catch (InvalidCastException e)
{
// Perform some action here, and then throw a new exception.
throw new YourCustomException("Put your error message here.", e);
}
Você pode também relançar uma exceção quando uma determinada condição for verdadeira, conforme
mostrado no exemplo a seguir.
catch (InvalidCastException e)
{
if (e.Data == null)
{
throw;
}
else
{
// Take some action.
}
}
NOTE
Também é possível usar um filtro de exceção para obter um resultado semelhante em um modo geralmente mais
limpo (além de não modificar a pilha, conforme explicado anteriormente neste documento). O exemplo a seguir tem
um comportamento semelhante para chamadores como no exemplo anterior. A função gera a
InvalidCastException novamente para o chamador quando e.Data é null .
De dentro de um bloco try , inicialize somente as variáveis que são declaradas nele. Caso contrário, uma
exceção pode ocorrer antes da conclusão da execução do bloco. Por exemplo, no exemplo de código a
seguir, a variável n é inicializada dentro do bloco try . Uma tentativa de usar essa variável fora do bloco
try na instrução Write(n) gerará um erro de compilador.
Exemplo
No exemplo a seguir, o bloco try contém uma chamada para o método ProcessString que pode causar
uma exceção. A cláusula catch contém o manipulador de exceção que apenas exibe uma mensagem na
tela. Quando instrução throw é chamada de dentro de MyMethod , o sistema procura a instrução catch e
exibe a mensagem Exception caught .
class TryFinallyTest
{
static void ProcessString(string s)
{
if (s == null)
{
throw new ArgumentNullException();
}
}
try
{
ProcessString(s);
}
catch (Exception e)
{
Console.WriteLine("{0} Exception caught.", e);
}
}
}
/*
Output:
System.ArgumentNullException: Value cannot be null.
at TryFinallyTest.Main() Exception caught.
* */
Remova a marca de comentário da linha throw new OperationCanceledException para demonstrar o que
acontece quando você cancela um processo assíncrono. A propriedade IsCanceled da tarefa é definida
para true e a exceção é capturada no bloco catch . Em algumas condições que não se aplicam a este
exemplo, a propriedade IsFaulted da tarefa é definida para true e IsCanceled é definido para false .
public async Task DoSomethingAsync()
{
Task<string> theTask = DelayAsync();
try
{
string result = await theTask;
Debug.WriteLine("Result: " + result);
}
catch (Exception ex)
{
Debug.WriteLine("Exception Message: " + ex.Message);
}
Debug.WriteLine("Task IsCanceled: " + theTask.IsCanceled);
Debug.WriteLine("Task IsFaulted: " + theTask.IsFaulted);
if (theTask.Exception != null)
{
Debug.WriteLine("Task Exception Message: "
+ theTask.Exception.Message);
Debug.WriteLine("Task Inner Exception Message: "
+ theTask.Exception.InnerException.Message);
}
}
Exemplo de Task.WhenAll
O exemplo a seguir ilustra a manipulação de exceção em que várias tarefas podem resultar em várias
exceções. O bloco try aguarda a tarefa que é retornada por uma chamada para Task.WhenAll. A tarefa é
concluída quando as três tarefas às quais WhenAll se aplica são concluídas.
Cada uma das três tarefas causa uma exceção. O bloco catch itera por meio de exceções, que são
encontradas na propriedade Exception.InnerExceptions da tarefa que foi retornada por Task.WhenAll.
public async Task DoMultipleAsync()
{
Task theTask1 = ExcAsync(info: "First Task");
Task theTask2 = ExcAsync(info: "Second Task");
Task theTask3 = ExcAsync(info: "Third Task");
try
{
await allTasks;
}
catch (Exception ex)
{
Debug.WriteLine("Exception: " + ex.Message);
Debug.WriteLine("Task IsFaulted: " + allTasks.IsFaulted);
foreach (var inEx in allTasks.Exception.InnerExceptions)
{
Debug.WriteLine("Task Inner Exception: " + inEx.Message);
}
}
}
// Output:
// Exception: Error-First Task
// Task IsFaulted: True
// Task Inner Exception: Error-First Task
// Task Inner Exception: Error-Second Task
// Task Inner Exception: Error-Third Task
Especificação da linguagem C#
Para obter mais informações, confira a seção A instrução try da Especificação da linguagem C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Instruções try, throw e catch (C++)
throw
try-finally
Como: Gerar exceções explicitamente
try-finally (Referência de C#)
23/10/2019 • 5 minutes to read • Edit Online
Usando um bloco finally , você pode limpar todos os recursos alocados em um bloco try e pode executar
código mesmo se uma exceção ocorrer no bloco try . Normalmente, as instruções de um bloco finally são
executadas quando o controle deixa uma instrução try . A transferência de controle pode ocorrer como
resultado da execução normal, da execução de uma instrução break , continue , goto ou return , ou da
propagação de uma exceção para fora da instrução try .
Dentro de uma exceção tratada, é garantido que o bloco finally será executado. No entanto, se a exceção
não for tratada, a execução do bloco finally depende de como a operação de desenrolamento da exceção é
disparada. Isso, por sua vez, depende da configuração do seu computador.
Normalmente, quando uma exceção sem tratamento encerra um aplicativo, não é importante se o bloco
finally é executado. No entanto, se você tiver instruções em um bloco finally que devem ser executadas
mesmo nesse caso, uma solução é adicionar um bloco catch à instrução try - finally . Como alternativa,
você pode detectar a exceção que pode ser gerada no bloco try de uma instrução try - finally em posição
superior na pilha de chamadas. Ou seja, você pode detectar a exceção no método que chama o método que
contém a instrução try - finally ou no método que chama esse método ou em qualquer método na pilha de
chamadas. Se a exceção não for detectada, a execução do bloco finally dependerá do sistema operacional
escolher disparar uma operação de desenrolamento de exceção.
Exemplo
No exemplo a seguir, uma instrução de conversão inválida causa uma exceção System.InvalidCastException .A
exceção não é tratada.
public class ThrowTestA
{
static void Main()
{
int i = 123;
string s = "Some string";
object obj = s;
try
{
// Invalid conversion; obj contains a string, not a numeric type.
i = (int)obj;
No exemplo a seguir, uma exceção do método TryCast ocorre em um método mais para cima na pilha de
chamadas.
public class ThrowTestB
{
static void Main()
{
try
{
// TryCast produces an unhandled exception.
TryCast();
}
catch (Exception ex)
{
// Catch the exception that is unhandled in TryCast.
Console.WriteLine
("Catching the {0} exception triggers the finally block.",
ex.GetType());
try
{
// Invalid conversion; obj contains a string, not a numeric type.
i = (int)obj;
Especificação da linguagem C#
Para obter mais informações, confira a seção A instrução try da Especificação da linguagem C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Instruções try, throw e catch (C++)
throw
try-catch
Como: Gerar exceções explicitamente
try-catch-finally (Referência de C#)
23/10/2019 • 2 minutes to read • Edit Online
Um uso comum de catch e finally juntos é obter e usar recursos em um bloco try , lidar com
circunstâncias excepcionais em um bloco catch e liberar os recursos no bloco finally .
Para obter mais informações e exemplos sobre como lançar exceções, consulte try-catch e Lançando exceções.
Para obter mais informações sobre o bloco finally , consulte try-finally.
Exemplo
public class EHClass
{
void ReadFile(int index)
{
// To run this code, substitute a valid path from your local machine
string path = @"c:\users\public\test.txt";
System.IO.StreamReader file = new System.IO.StreamReader(path);
char[] buffer = new char[10];
try
{
file.ReadBlock(buffer, index, buffer.Length);
}
catch (System.IO.IOException e)
{
Console.WriteLine("Error reading from {0}. Message = {1}", path, e.Message);
}
finally
{
if (file != null)
{
file.Close();
}
}
// Do something with buffer...
}
Especificação da linguagem C#
Para obter mais informações, confira a seção A instrução try da Especificação da linguagem C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Instruções try, throw e catch (C++)
throw
Como: Gerar exceções explicitamente
Instrução using
Contexto verificado e não verificado (Referência de
C#)
24/10/2019 • 2 minutes to read • Edit Online
Conversões numéricas explícitas entre tipos integrais ou de float ou double para um tipo integral.
Se nem checked ou unchecked for especificado, o contexto padrão de expressões de não constante (expressões
que são avaliadas no tempo de execução) é definido pelo valor da opção do compilador -checked. Por padrão, o
valor dessa opção é removido e as operações aritméticas são executadas em um contexto não verificado.
Para expressões de constante (expressões que podem ser totalmente avaliadas no tempo de compilação), o
contexto padrão sempre é verificado. A menos que uma expressão de constante seja explicitamente colocada
em um contexto não verificado, estouros que ocorrem durante a avaliação do tempo de compilação da
expressão causam erros de tempo de compilação.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Palavras-chave de instrução
checked (Referência de C#)
23/10/2019 • 3 minutes to read • Edit Online
A palavra-chave checked é usada para habilitar explicitamente a verificação estouro para conversões e
operações aritméticas de tipo integral.
Por padrão, uma expressão que contém somente valores constantes causa um erro do compilador se a
expressão produzir um valor fora do intervalo do tipo de destino. Se a expressão contiver um ou mais valores
não constantes, o compilador não detectará o estouro. Avaliar a expressão atribuída a i2 no exemplo a seguir
não causa um erro do compilador.
// The following example, which includes variable ten, does not cause
// a compiler error.
int ten = 10;
int i2 = 2147483647 + ten;
Por padrão, essas expressões não constantes não são verificados quanto ao estouro em tempo de execução e
não geram exceções de estouro. O exemplo anterior exibe -2,147,483,639 como a soma de dois números
inteiros positivos.
A verificação de estouro pode ser habilitada por opções do compilador, configuração do ambiente ou uso da
palavra-chave checked . Os exemplos a seguir demonstram como usar uma expressão checked ou um bloco
checked para detectar o estouro produzido pela soma anterior no tempo de execução. Os dois exemplos geram
uma exceção de estouro.
// Checked expression.
Console.WriteLine(checked(2147483647 + ten));
// Checked block.
checked
{
int i3 = 2147483647 + ten;
Console.WriteLine(i3);
}
Exemplo
Este exemplo mostra como usar checked para habilitar a verificação de estouro em tempo de execução.
class OverFlowTest
{
// Set maxIntValue to the maximum value for integers.
static int maxIntValue = 2147483647;
Especificação da linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a fonte
definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Checked e Unchecked
unchecked
unchecked (Referência de C#)
23/10/2019 • 3 minutes to read • Edit Online
A palavra-chave unchecked é usada para suprimir a verificação estouro para conversões e operações
aritméticas de tipo integral.
Em um contexto não verificado, se uma expressão produzir um valor que está fora do intervalo do tipo de
destino, o estouro não será sinalizado. Por exemplo, como o cálculo no exemplo a seguir é realizado em uma
expressão ou bloco unchecked , o fato de o resultado ser muito grande para um inteiro é ignorado e int1
recebe a atribuição do valor -2.147.483.639.
unchecked
{
int1 = 2147483647 + 10;
}
int1 = unchecked(ConstantMax + 10);
Se o ambiente unchecked for removido, ocorrerá um erro de compilação. O estouro pode ser detectado em
tempo de compilação porque todos os termos da expressão são constantes.
Expressões que contêm termos não constantes não são verificadas por padrão em tempo de compilação e
tempo de execução. Consulte checked para obter informações sobre como habilitar um ambiente verificado.
Como a verificação de estouro leva tempo, o uso de código não verificado em situações em que não há
nenhum risco de estouro pode melhorar o desempenho. No entanto, se o estouro for uma possibilidade, um
ambiente verificado deverá ser usado.
Exemplo
Esse exemplo mostra como usar a palavra-chave unchecked .
class UncheckedDemo
{
static void Main(string[] args)
{
// int.MaxValue is 2,147,483,647.
const int ConstantMax = int.MaxValue;
int int1;
int int2;
int variableMax = 2147483647;
// The following statements are checked by default at compile time. They do not
// compile.
//int1 = 2147483647 + 10;
//int1 = ConstantMax + 10;
// To enable the assignments to int1 to compile and run, place them inside
// an unchecked block or expression. The following statements compile and
// run.
unchecked
{
int1 = 2147483647 + 10;
}
int1 = unchecked(ConstantMax + 10);
// To catch the overflow in the assignment to int2 at run time, put the
// declaration in a checked block or expression. The following
// statements compile but raise an overflow exception at run time.
checked
{
//int2 = variableMax + 10;
}
//int2 = checked(variableMax + 10);
Especificação da linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a fonte
definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Checked e Unchecked
checked
Instrução fixed (Referência de C#)
23/10/2019 • 5 minutes to read • Edit Online
A instrução fixed impede que o coletor de lixo faça a realocação de uma variável móvel. A instrução fixed
é permitida somente em um contexto não seguro. Você também pode usar a palavra-chave fixed para criar
buffers de tamanho fixo.
A instrução fixed define um ponteiro para uma variável gerenciada e "fixa" essa variável durante a
execução da instrução. Os ponteiros móveis gerenciados são úteis apenas em um contexto fixed . Sem um
contexto fixed , a coleta de lixo poderia realocar as variáveis de forma imprevisível. O compilador do C# só
permite que você atribua um ponteiro a uma variável gerenciada em uma instrução fixed .
class Point
{
public int x;
public int y;
}
Você pode inicializar um ponteiro usando uma matriz, uma cadeia de caracteres, um buffer de tamanho fixo
ou o endereço de uma variável. O exemplo a seguir ilustra o uso de endereços de variáveis, matrizes e
sequências de caracteres:
// The following two assignments are equivalent. Each assigns the address
// of the first element in array arr to pointer p.
Se você estiver criando tipos que devem participar desse padrão, consulte Span<T>.GetPinnableReference()
para obter um exemplo da implementação do padrão.
Vários ponteiros podem ser inicializados em uma instrução quando eles são do mesmo tipo:
Para inicializar ponteiros de tipos diferentes, basta aninhar instruções fixed , conforme mostrado no
exemplo a seguir.
Depois que o código na instrução é executado, todas as variáveis fixadas são desafixadas e ficam sujeiras à
coleta de lixo. Portanto, não aponte para essas variáveis fora da instrução fixed . As variáveis declaradas na
instrução fixed estão no escopo dessa instrução, facilitando isto:
É possível alocar memória na pilha, local que não está sujeito à coleta de lixo e, portanto, não precisa ser
fixado. Para fazer isso, use o operador stackalloc .
Especificação da linguagem C#
Para saber mais, confira a seção A instrução corrigida na Especificação da linguagem C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
unsafe
Tipos de ponteiro
Buffers de tamanho fixo
instrução lock (referência em C#)
23/10/2019 • 4 minutes to read • Edit Online
A instrução lock obtém o bloqueio de exclusão mútua para um determinado objeto, executa um bloco de
instruções e, em seguida, libera o bloqueio. Embora um bloqueio seja mantido, o thread que mantém o bloqueio
pode adquiri-lo novamente e liberá-lo. Qualquer outro thread é impedido de adquirir o bloqueio e aguarda até
que ele seja liberado.
A instrução lock está no formato
lock (x)
{
// Your code...
}
object __lockObj = x;
bool __lockWasTaken = false;
try
{
System.Threading.Monitor.Enter(__lockObj, ref __lockWasTaken);
// Your code...
}
finally
{
if (__lockWasTaken) System.Threading.Monitor.Exit(__lockObj);
}
Como o código usa um bloco try...finally, o bloqueio será liberado mesmo se uma exceção for gerada dentro do
corpo de uma instrução lock .
Não é possível usar o operador await no corpo de uma instrução lock .
Comentários
Ao sincronizar o acesso de thread com um recurso compartilhado, bloqueie uma instância de objeto dedicada
(por exemplo, private readonly object balanceLock = new object(); ) ou outra instância que provavelmente não
será usada como um objeto de bloqueio por partes não relacionadas do código. Evite usar a mesma instância de
objeto de bloqueio para diferentes recursos compartilhados, uma vez que ela poderia resultar em deadlock ou
contenção de bloqueio. Especificamente, evite usar os seguintes itens como objetos de bloqueio:
this , uma vez que pode ser usado pelos chamadores como um bloqueio.
Instâncias Type, pois podem ser obtidas pelo operador ou reflexão typeof.
Instâncias de cadeia de caracteres, incluindo literais de cadeia de caracteres, pois podem ser internalizadas.
Exemplo
O exemplo a seguir define uma classe Account que sincroniza o acesso com seu campo privado balance
bloqueando uma instância balanceLock dedicada. Usar a mesma instância para bloquear garante que o campo
balance não pode ser atualizado simultaneamente por dois threads que tentam chamar os métodos Debit ou
Credit simultaneamente.
using System;
using System.Threading.Tasks;
class AccountTest
{
static void Main()
{
var account = new Account(1000);
var tasks = new Task[100];
for (int i = 0; i < tasks.Length; i++)
{
tasks[i] = Task.Run(() => RandomlyUpdate(account));
}
Task.WaitAll(tasks);
}
Especificação da linguagem C#
Para saber mais, confira a seção A instrução lock na especificação da linguagem C#.
Consulte também
System.Threading.Monitor
System.Threading.SpinLock
System.Threading.Interlocked
Referência de C#
Palavras-chave do C#
Visão geral dos primitivos de sincronização
Parâmetros de método (Referência de C#)
23/10/2019 • 2 minutes to read • Edit Online
Os parâmetros declarados para um método sem in, ref nem out são passados para o método chamado pelo valor.
Esse valor pode ser alterado no método, mas o valor alterado não será mantido quando o controle passá-lo de
volta para o procedimento de chamada. É possível alterar esse comportamento usando uma palavra-chave de
parâmetro de método.
Esta seção descreve as palavras-chave que podem ser usadas ao declarar parâmetros de método:
params especifica que esse parâmetro pode receber um número variável de argumentos.
in especifica que esse parâmetro é passado por referência, mas é lido apenas pelo método chamado.
ref especifica que esse parâmetro é passado por referência e pode ser lido ou gravado pelo método
chamado.
out especifica que esse parâmetro é passado por referência e é gravado pelo método chamado.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
params (Referência de C#)
23/10/2019 • 2 minutes to read • Edit Online
Usando a palavra-chave params , você pode especificar um parâmetro do método que aceita um número variável
de argumentos.
Você pode enviar uma lista separada por vírgulas dos argumentos do tipo especificado na declaração de
parâmetros ou uma matriz de argumentos do tipo especificado. Você também pode não enviar nenhum
argumento. Se você não enviar nenhum argumento, o comprimento da lista params será zero.
Nenhum parâmetro adicional é permitido após a palavra-chave params em uma declaração de método e apenas
uma palavra-chave params é permitida em uma declaração de método.
O tipo declarado do parâmetro params precisa ser uma matriz unidimensional, como mostra o exemplo a seguir.
Caso contrário, ocorrerá o erro do compilador CS0225.
Exemplo
O exemplo a seguir demonstra várias maneiras em que os argumentos podem ser enviados para um parâmetro
params .
public class MyClass
{
public static void UseParams(params int[] list)
{
for (int i = 0; i < list.Length; i++)
{
Console.Write(list[i] + " ");
}
Console.WriteLine();
}
// The following call does not cause an error, but the entire
// integer array becomes the first element of the params array.
UseParams2(myIntArray);
}
}
/*
Output:
1 2 3 4
1 a test
5 6 7 8 9
2 b test again
System.Int32[]
*/
Especificação da linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a fonte
definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Parâmetros de método
Modificador de parâmetro in (referência do C#)
23/10/2019 • 9 minutes to read • Edit Online
A palavra-chave in faz com que os argumentos sejam passados por referência. Ela torna o parâmetro
formal um alias para o argumento, que deve ser uma variável. Em outras palavras, qualquer operação no
parâmetro é feita no argumento. É como as palavras-chave ref ou out, exceto que os argumentos in não
podem ser modificados pelo método chamado. Enquanto os argumentos ref podem ser modificados, os
argumentos out devem ser modificados pelo método chamado, e essas modificações podem ser
observadas no contexto da chamada.
NOTE
A palavra-chave in também pode ser usada com um parâmetro de tipo genérico para especificar que o
parâmetro de tipo é contravariante, como parte de uma instrução foreach ou como parte de uma cláusula
join em uma consulta LINQ. Para obter mais informações sobre o uso da palavra-chave in nesses contextos,
confira in que fornece links para todos esses usos.
As variáveis passadas como argumentos in precisam ser inicializadas antes de serem passadas em uma
chamada de método. No entanto, o método chamado não pode atribuir um valor nem modificar o
argumento.
O modificador de parâmetro in está disponível no C# 7.2 e posteriores. As versões anteriores geravam o
erro de compilador CS8107 ("o recurso 'referências somente leitura' não está disponível no C# 7.0. Use a
linguagem na versão 7.2 ou posteriores"). Confira como configurar a versão de linguagem do compilador
em Selecionar a versão da linguagem C#.
As palavras-chave in , ref e out não são consideradas parte da assinatura do método para fins de
resolução de sobrecarga. Portanto, os métodos não poderão ser sobrecarregados se a única diferença for
que um método usa um argumento ref ou in e o outro usa um argumento out . Por exemplo, o
código a seguir, não será compilado:
class CS0663_Example
{
// Compiler error CS0663: "Cannot define overloaded
// methods that differ only on in, ref and out".
public void SampleMethod(in int i) { }
public void SampleMethod(ref int i) { }
}
A sobrecarga com base na presença de in é permitida:
class InOverloads
{
public void SampleMethod(in int i) { }
public void SampleMethod(int i) { }
}
Segundo: especificar in declara sua intenção de passar um argumento por referência. O argumento
usado com in deve representar um local ao qual se possa fazer referência diretamente. As mesmas
regras gerais para argumentos out e ref se aplicam: você não pode usar constantes, propriedades
comuns ou outras expressões que produzem valores. Caso contrário, a omissão de in no site de
chamada informa ao compilador que você permitirá a criação de uma variável temporária para passar por
referência de somente leitura para o método. O compilador cria uma variável temporária para superar
várias restrições com argumentos in :
Uma variável temporária permite constantes de tempo de compilação como parâmetros in .
Uma variável temporária permite propriedades ou outras expressões para parâmetros in .
Uma variável temporária permite argumentos em que há uma conversão implícita do tipo de
argumento para o tipo de parâmetro.
Em todas as instâncias anteriores, o compilador cria uma variável temporária que armazena o valor da
constante, da propriedade ou de outra expressão.
O código a seguir ilustra essas regras:
static void Method(in int argument)
{
// implementation removed
}
Agora, vamos supor que outro método usando argumentos por valor estivesse disponível. Os resultados
são alterados conforme mostrado no código a seguir:
A única chamada de método em que o argumento é passado por referência é a chamada final.
NOTE
O código anterior usa int como o tipo de argumento por questão de simplicidade. Como int não é maior que
uma referência na maioria dos computadores modernos, não há nenhum benefício em passar um único int como
uma referência readonly.
Limitações em parâmetros in
Não é possível usar as palavras-chave in , ref e out para os seguintes tipos de métodos:
Métodos assíncronos, que você define usando o modificador async.
Métodos de iterador, que incluem uma instrução yield return ou yield break .
Especificação da Linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a
fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Parâmetros de método
Escrever código eficiente seguro
ref (Referência de C#)
23/10/2019 • 17 minutes to read • Edit Online
A palavra-chave ref indica um valor que é passado por referência. Ela é usada em quatro contextos
diferentes:
Em uma assinatura de método e em uma chamada de método, para passar um argumento a um
método por referência. Para obter mais informações, veja Passar um argumento por referência.
Em uma assinatura de método para retornar um valor para o chamador por referência. Para obter
mais informações, consulte Reference return values (Valores retornados de referência).
Em um corpo de membro, para indicar que um valor retornado por referência é armazenado
localmente como uma referência que o chamador pretende modificar ou, em geral, uma variável
local acessa outro valor por referência. Para obter mais informações, veja Locais de referência.
Em uma declaração struct para declarar um ref struct ou um readonly ref struct . Para obter
mais informações, veja tipos ref struct.
NOTE
Não confunda o conceito de passar por referência com o conceito de tipos de referência. Os dois conceitos
não são iguais. Um parâmetro de método pode ser modificado por ref , independentemente de ele ser um
tipo de valor ou um tipo de referência. Não há nenhuma conversão boxing de um tipo de valor quando ele é
passado por referência.
Para usar um parâmetro ref , a definição do método e o método de chamada devem usar
explicitamente a palavra-chave ref , como mostrado no exemplo a seguir.
int number = 1;
Method(ref number);
Console.WriteLine(number);
// Output: 45
Um argumento passado para um parâmetro ref ou in precisa ser inicializado antes de ser
passado. Isso é diferente dos parâmetros out, cujos argumentos não precisam ser inicializados
explicitamente antes de serem passados.
Os membros de uma classe não podem ter assinaturas que se diferem somente por ref , in ou
out . Ocorrerá um erro de compilador se a única diferença entre os dois membros de um tipo for
que um deles tem um parâmetro ref e o outro tem um parâmetro out ou in . O código a seguir,
por exemplo, não é compilado.
class CS0663_Example
{
// Compiler error CS0663: "Cannot define overloaded
// methods that differ only on ref and out".
public void SampleMethod(out int i) { }
public void SampleMethod(ref int i) { }
}
No entanto, os métodos podem ser sobrecarregados quando um método tem um parâmetro ref ,
in ou out e o outro tem um parâmetro de valor, conforme é mostrado no exemplo a seguir.
class RefOverloadExample
{
public void SampleMethod(int i) { }
public void SampleMethod(ref int i) { }
}
Em outras situações que exigem correspondência de assinatura, como ocultar ou substituir, in , ref
e out fazem parte da assinatura e não são correspondentes.
Propriedades não são variáveis. Elas são métodos e não podem ser passadas para parâmetros ref .
Não é possível usar as palavras-chave ref , in e out para os seguintes tipos de métodos:
Métodos assíncronos, que você define usando o modificador async.
Métodos de iterador, que incluem uma instrução yield return ou yield break .
Para obter mais informações sobre como passar tipos de referência por valor e por referência,
consulte Passando parâmetros de tipo de referência.
Entre o token return e a variável retornada em uma instrução return no método. Por exemplo:
Ref locals
Uma variável de ref local é usada para fazer referência a valores retornados usando return ref . Uma
variável local ref não pode ser inicializada para um valor retornado que não seja ref. Em outras
palavras, o lado direito da inicialização deve ser uma referência. Todas as modificações ao valor do ref
local são refletidas no estado do objeto cujo método retornou o valor por referência.
Você define um ref local usando a palavra-chave ref antes da declaração de variável, bem como
imediatamente antes da chamada para o método que retorna o valor por referência.
Por exemplo, a instrução a seguir define um valor de ref local que é retornado por um método
chamado GetEstimatedValue :
Você pode acessar um valor por referência da mesma maneira. Em alguns casos, acessar um valor
por referência aumenta o desempenho, evitando uma operação de cópia potencialmente dispendiosa.
Por exemplo, a instrução a seguir mostra como é possível definir um valor de local de ref que é usado
para fazer referência a um valor.
Observe que, nos dois exemplos, a palavra-chave ref deve ser usada em ambos os locais ou o
compilador gera o erro CS8172, "Não é possível inicializar uma variável por referência com um
valor".
A partir do C# 7.3, a variável de iteração da instrução foreach pode ser ref local ou a variável local
ref readonly. Para saber mais, confira o artigo Instrução foreach.
Quando o chamador armazena o valor retornado pelo método GetBookByTitle como um ref local, as
alterações que o chamador faz ao valor retornado são refletidas no objeto BookCollection , conforme
mostra o exemplo a seguir.
Essas restrições garantem que você não use acidentalmente um ref struct de maneira que possa
promovê-lo para o heap gerenciado.
Você pode combinar modificadores para declarar um struct como readonly ref . Um
readonly ref struct combina os benefícios e as restrições de declarações ref struct e
readonly struct .
Especificação da linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da
linguagem é a fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Escrever código eficiente seguro
Retornos de ref e locais de ref
Expressão condicional ref
Operador de atribuição ref
Passando parâmetros
Parâmetros de método
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Modificador de parâmetro out (Referência de
C#)
23/10/2019 • 6 minutes to read • Edit Online
A palavra-chave out faz com que os argumentos sejam passados por referência. Ela torna o
parâmetro formal um alias para o argumento, que deve ser uma variável. Em outras palavras, qualquer
operação no parâmetro é feita no argumento. É como a palavra-chave ref, exceto pelo fato de que ref
requer que a variável seja inicializada antes de ser passada. Também é como a palavra-chave in, exceto
que in não permite que o método chamado modifique o valor do argumento. Para usar um
parâmetro out , a definição do método e o método de chamada devem usar explicitamente a palavra-
chave out . Por exemplo:
int initializeInMethod;
OutArgExample(out initializeInMethod);
Console.WriteLine(initializeInMethod); // value is now 44
NOTE
A palavra-chave out também pode ser usada com um parâmetro de tipo genérico para especificar que o
parâmetro de tipo é covariante. Para obter mais informações sobre o uso da palavra-chave out nesse
contexto, consulte out (modificador genérico).
Variáveis passadas como argumentos out não precisam ser inicializadas antes de serem passadas em
uma chamada de método. No entanto, o método chamado é necessário para atribuir um valor antes
que o método seja retornado.
As palavras-chave in , ref e out não são consideradas parte da assinatura do método para fins de
resolução de sobrecarga. Portanto, os métodos não poderão ser sobrecarregados se a única diferença
for que um método usa um argumento ref ou in e o outro usa um argumento out . Por exemplo, o
código a seguir, não será compilado:
class CS0663_Example
{
// Compiler error CS0663: "Cannot define overloaded
// methods that differ only on ref and out".
public void SampleMethod(out int i) { }
public void SampleMethod(ref int i) { }
}
A sobrecarga será válida, no entanto, se um método usar um argumento ref , in ou out e o outro
não tiver nenhum desses modificadores, desta forma:
class OutOverloadExample
{
public void SampleMethod(int i) { }
public void SampleMethod(out int i) => i = 5;
}
void Method(out int answer, out string message, out string stillNull)
{
answer = 44;
message = "I've been returned";
stillNull = null;
}
int argNumber;
string argMessage, argDefault;
Method(out argNumber, out argMessage, out argDefault);
Console.WriteLine(argNumber);
Console.WriteLine(argMessage);
Console.WriteLine(argDefault == null);
int number;
if (Int32.TryParse(numberAsString, out number))
Console.WriteLine($"Converted '{numberAsString}' to {number}");
else
Console.WriteLine($"Unable to convert '{numberAsString}'");
// The example displays the following output:
// Converted '1640' to 1640
Começando com o C#7.0, você pode declarar a variável out na lista de argumentos da chamada de
método em vez de declará-la em uma declaração de variável separada. Isso produz um código mais
compacto e legível, além de impedir que você atribua acidentalmente um valor à variável antes da
chamada de método. O exemplo a seguir é semelhante ao exemplo anterior, exceto por definir a
variável number na chamada para o método Int32.TryParse.
No exemplo anterior, a variável number é fortemente tipada como um int . Você também pode
declarar uma variável local de tipo implícito, como no exemplo a seguir.
Especificação da Linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem
é a fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Parâmetros de método
namespace (Referência de C#)
04/11/2019 • 2 minutes to read • Edit Online
A palavra-chave namespace é usada para declarar um escopo que contém um conjunto de objetos relacionados.
Você pode usar um namespace para organizar elementos de código e criar tipos globalmente exclusivos.
namespace SampleNamespace
{
class SampleClass { }
interface ISampleInterface { }
struct SampleStruct { }
enum SampleEnum { a, b }
namespace Nested
{
class SampleClass2 { }
}
}
Comentários
Dentro de um namespace, é possível declarar zero ou mais dos seguintes tipos:
outro namespace
class
interface
struct
enum
delegate
Quer você declare explicitamente ou não um namespace em um arquivo de origem C#, o compilador adiciona
um namespace padrão. Este namespace sem nome, às vezes chamado de namespace global, está presente em
todos os arquivos. Qualquer identificador no namespace global está disponível para uso em um namespace
nomeado.
Os namespaces implicitamente têm acesso público e não isso é modificável. Para uma discussão sobre os
modificadores de acesso que você pode atribuir a elementos em um namespace, consulte Modificadores de
acesso.
É possível definir um namespace em duas ou mais declarações. Por exemplo, o exemplo a seguir define duas
classes como parte do namespace MyCompany :
namespace MyCompany.Proj1
{
class MyClass
{
}
}
namespace MyCompany.Proj1
{
class MyClass1
{
}
}
Exemplo
O exemplo a seguir mostra como chamar um método estático em um namespace aninhado.
namespace SomeNameSpace
{
public class MyClass
{
static void Main()
{
Nested.NestedNameSpaceClass.SayHello();
}
}
// a nested namespace
namespace Nested
{
public class NestedNameSpaceClass
{
public static void SayHello()
{
Console.WriteLine("Hello");
}
}
}
}
// Output: Hello
Especificação da linguagem C#
Para saber mais, confira a seção Namespaces da Especificação da linguagem C#.
Consulte também
Referência de C#
Palavras-chave do C#
using
using static
Qualificador de alias de namespace ::
Namespaces
using (Referência de C#)
23/10/2019 • 2 minutes to read • Edit Online
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Namespaces
extern
Diretiva using (Referência de C#)
04/11/2019 • 5 minutes to read • Edit Online
using System.Text;
Para permitir que você acesse membros estáticos e tipos aninhados de um tipo sem precisar qualificar
o acesso com o nome do tipo.
A palavra-chave using também é usada para criar instruções using, o que ajuda a garantir que objetos
IDisposable, tais como arquivos e fontes, sejam tratados corretamente. Consulte a Instrução using para obter
mais informações.
Comentários
O escopo de uma diretiva using é limitado ao arquivo em que ele aparece.
A diretiva using pode aparecer:
No início de um arquivo de código-fonte, antes de quaisquer definições de namespace ou tipo.
Em qualquer namespace, mas antes de qualquer namespace ou tipos declarados neste namespace.
Caso contrário, serão gerados erros do compilador CS1529.
Crie uma diretiva de alias using para tornar mais fácil a qualificação de um identificador para um
namespace ou tipo. Em qualquer diretiva using , o namespace totalmente qualificado ou o tipo deve ser
usado independentemente das diretivas using que vêm antes. Nenhum alias using pode ser usado na
declaração de uma diretiva using . Por exemplo, o código a seguir gera um erro de compilador:
using s = System.Text;
using s.RegularExpressions;
Crie uma diretiva using para usar os tipos em um namespace sem precisar especificar o namespace. Uma
diretiva using não fornece acesso a nenhum namespace aninhado no namespace especificado.
Os namespaces vêm em duas categorias: definidos pelo usuário e definidos pelo sistema. Os namespaces
definidos pelo usuário são namespaces definidos em seu código. Para obter uma lista dos namespaces
definidos pelo sistema, consulte Navegador de API do .NET.
Exemplo 1
O exemplo a seguir mostra como definir e usar um alias de using para um namespace:
namespace PC
{
// Define an alias for the nested namespace.
using Project = PC.MyCompany.Project;
class A
{
void M()
{
// Use the alias
var mc = new Project.MyClass();
}
}
namespace MyCompany
{
namespace Project
{
public class MyClass { }
}
}
}
Uma diretiva alias de using não pode ter um tipo genérico aberto no lado direito. Por exemplo, você não
pode criar um alias de using para um List<T> , mas pode criar um para um List<int> .
Exemplo 2
O exemplo a seguir mostra como definir uma diretiva using e um alias using para uma classe:
using System;
namespace NameSpace1
{
public class MyClass
{
public override string ToString()
{
return "You are in NameSpace1.MyClass.";
}
}
namespace NameSpace2
{
class MyClass<T>
{
public override string ToString()
{
return "You are in NameSpace2.MyClass.";
}
}
}
namespace NameSpace3
{
class MainClass
{
static void Main()
{
var instance1 = new AliasToMyClass();
Console.WriteLine(instance1);
}
}
}
// Output:
// You are in NameSpace1.MyClass.
// You are in NameSpace2.MyClass.
Especificação da linguagem C#
Para obter mais informações, consulte Diretivas using na Especificação da Linguagem C#. A especificação da
linguagem é a fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
Usando namespaces
Palavras-chave do C#
Namespaces
Instrução using
Diretiva using static (Referência de C#)
31/10/2019 • 6 minutes to read • Edit Online
A diretiva using static designa um tipo cujos membros estáticos e tipos aninhados você pode acessar sem
especificar um nome de tipo. A sintaxe é:
em que fully-qualified -type-name é o nome do tipo cujos membros estáticos e tipos aninhados podem ser
referenciados sem especificar um nome de tipo. Se você não fornecer um nome de tipo totalmente qualificado (o
nome do namespace completo juntamente com o nome do tipo), o C# gerará o erro de compilador CS0246: “O
tipo ou o nome do namespace 'type/namespace' não pôde ser encontrado (uma diretiva using ou uma referência
de assembly está ausente?)”.
A diretiva using static aplica-se a qualquer tipo que tenha membros estático (ou tipos aninhados), mesmo que
ele também tenha membros de instância. No entanto, os membros da instância podem ser invocados apenas por
meio de instância de tipo.
A diretiva using static foi introduzida no C# 6.
Comentários
Normalmente, quando você chamar um membro estático, fornece o nome do tipo juntamente com o nome do
membro. Inserir repetidamente o mesmo nome de tipo para invocar os membros do tipo pode resultar em
código obscuro detalhado. Por exemplo, a seguinte definição de uma classe Circle faz referência a um número
de membros da classe Math.
using System;
using System;
using static System.Math;
using static importa somente os membros estáticos acessíveis e os tipos aninhados declarados no tipo
especificado. Os membros herdados não são importados. Você pode importar de qualquer tipo nomeado com
uma diretiva using static, incluindo módulos do Visual Basic. Se funções de nível superior do F# aparecerem nos
metadados como membros estáticos de um tipo nomeado cujo nome é um identificador válido do C#, as
funções do F# poderão ser importadas.
using static torna os métodos de extensão declarados no tipo especificado disponível para pesquisa de
método de extensão. No entanto, os nomes dos métodos de extensão não são importados no escopo para a
referência não qualificada no código.
Métodos com o mesmo nome importados de diferentes tipos por diferentes diretivas using static na mesma
unidade de compilação ou namespace formam um grupo de métodos. A resolução de sobrecarga nesses grupos
de métodos segue as regras normais de C#.
Exemplo
O exemplo a seguir usa a diretiva using static para tornar os membros estáticos das classes Console, Math e
String disponíveis sem a necessidade de especificar seu nome de tipo.
using System;
using static System.Console;
using static System.Math;
using static System.String;
class Program
{
static void Main()
{
Write("Enter a circle's radius: ");
var input = ReadLine();
if (!IsNullOrEmpty(input) && double.TryParse(input, out var radius)) {
var c = new Circle(radius);
No exemplo, a diretiva using static também poderia ter sido aplicada ao tipo Double. Isso tornaria possível
chamar o método TryParse(String, Double) sem especificar um nome de tipo. No entanto, isso cria código menos
legível, uma vez que se torna necessário verificar as instruções using static para determinar qual método
TryParse do tipo numérico é chamado.
Consulte também
Diretiva using
Referência de C#
Palavras-chave do C#
Usando namespaces
Namespaces
Instrução using (Referência de C#)
04/11/2019 • 5 minutes to read • Edit Online
Fornece uma sintaxe conveniente que garante o uso correto de objetos IDisposable.
Exemplo
O exemplo a seguir mostra como usar a instrução using .
A partir C# do 8,0, você pode usar a seguinte sintaxe alternativa para a instrução de using que não exige
chaves:
Comentários
File e Font são exemplos de tipos gerenciados que acessam recursos não gerenciados (nesse caso,
identificadores de arquivo de caso e contextos de dispositivo). Há muitos outros tipos de recursos não
gerenciados e tipos de biblioteca de classes que os encapsula. Todos esses tipos devem implementar a
interface IDisposable.
Quando o tempo de vida de um objeto IDisposable é limitado a um único método, você deve declará-lo e
instanciá-lo na instrução using . A instrução using chama o método Dispose no objeto da forma correta e
(quando você o usa como mostrado anteriormente) ele também faz com que o objeto em si saia do escopo
assim que Dispose é chamado. Dentro do bloco using , o objeto é somente leitura e não pode ser
modificado ou reatribuído.
A instrução using garante que Dispose seja chamado, mesmo que ocorra uma exceção dentro do bloco
using . Você pode obter o mesmo resultado colocando o objeto dentro de um bloco try e então chamando
Dispose em um bloco finally . Na verdade, é dessa forma que a instrução using é convertida pelo
compilador. O exemplo de código anterior se expande para o seguinte código em tempo de compilação
(observe as chaves extras para criar o escopo limitado para o objeto):
{
var font1 = new Font("Arial", 10.0f);
try
{
byte charset = font1.GdiCharSet;
}
finally
{
if (font1 != null)
((IDisposable)font1).Dispose();
}
}
A sintaxe de instrução using mais recente se traduz em um código muito semelhante. O bloco de try é
aberto onde a variável é declarada. O bloco de finally é adicionado ao fechamento do bloco delimitador,
normalmente no final de um método.
Para obter mais informações sobre a instrução try - finally , veja o tópico try-finally.
Várias instâncias de um tipo podem ser declaradas na instrução using , conforme mostra o exemplo a seguir:
Você pode combinar várias declarações do mesmo tipo usando a nova sintaxe introduzida com C# 8
também. Isso é mostrado no exemplo a seguir:
Você pode criar uma instância do objeto de recurso e, em seguida, passar a variável para a instrução using ,
mas isso não é uma prática recomendada. Nesse caso, após o controle sair do bloco using , o objeto
permanecerá no escopo, mas provavelmente não terá acesso a seus recursos não gerenciados. Em outras
palavras, ele não é mais totalmente inicializado. Se você tentar usar o objeto fora do bloco using , corre o
risco de causar o lançamento de uma exceção. Por esse motivo, geralmente é melhor instanciar o objeto na
instrução using e limitar seu escopo ao bloco using .
Para obter mais informações sobre como descartar objetos IDisposable , veja Usando objetos que
implementam IDisposable.
Especificação da linguagem C#
Para obter mais informações, consulte A instrução using na Especificação da linguagem C#. A especificação
da linguagem é a fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Diretiva using
Coleta de lixo
Usando objetos que implementam IDisposable
Interface IDisposable
usando a instrução C# em 8,0
extern alias (Referência de C#)
23/10/2019 • 2 minutes to read • Edit Online
Talvez seja necessário referenciar duas versões de assemblies que têm os mesmos nomes de tipo totalmente
qualificado. Por exemplo, você pode ter que usar duas ou mais versões de um assembly no mesmo aplicativo. Ao
usar um alias de assembly externo, os namespaces de cada assembly podem ser encapsulados dentro de
namespaces de nível raiz nomeados pelo alias, permitindo que eles sejam utilizados no mesmo arquivo.
NOTE
A palavra-chave extern também é usada como um modificador de método, declarando um método escrito em código não
gerenciado.
Para referenciar dois assemblies com os mesmos nomes de tipo totalmente qualificado, um alias deve ser
especificado em um prompt de comando, da seguinte maneira:
/r:GridV1=grid.dll
/r:GridV2=grid20.dll
Isso cria os alias externos GridV1 e GridV2 . Para usar esses aliases de dentro de um programa, referencie-os
usando a palavra-chave extern . Por exemplo:
extern alias GridV1;
Cada declaração de alias externo apresenta um namespace de nível raiz adicional que funciona de forma paralela
(mas não dentro) com o namespace global. Portanto, os tipos de cada assembly podem ser referenciados sem
ambiguidade, usando seus nomes totalmente qualificados, enraizados no alias de namespace apropriado.
No exemplo anterior, GridV1::Grid seria o controle de grade da grid.dll e GridV2::Grid seria o controle de
grade da grid20.dll .
Especificação da Linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a fonte
definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Operador ::
-reference (opções do compilador do C#)
is (Referência de C#)
31/10/2019 • 12 minutes to read • Edit Online
O operador is verifica se o resultado de uma expressão é compatível com um tipo fornecido ou (a partir do
C# 7.0) testa uma expressão em relação a um padrão. Saiba mais sobre o operador is de teste de tipo na
seção operador is do artigo Operadores cast e teste de tipo.
Padrão de tipo, que testa se uma expressão pode ser convertida em um tipo especificado e, caso possa, a
converte em uma variável desse tipo.
Padrão constante, que testa se uma expressão é avaliada como um valor constante especificado.
Padrão var, uma correspondência que sempre é bem-sucedida e vincula o valor de uma expressão a uma
nova variável local.
Padrão de tipo
Ao usar o padrão de tipo para realizar a correspondência de padrões, is testa se uma expressão pode ser
convertida em um tipo especificado e, caso possa, a converte em uma variável desse tipo. Trata-se de uma
extensão simples da instrução is que habilita a conversão e a avaliação de tipo concisas. A forma geral do
padrão de tipo is é:
em que expr é uma expressão que é avaliada como uma instância de algum tipo, type é o nome do tipo no qual
o resultado de expr será convertido e varname é o objeto no qual o resultado de expr será convertido se o teste
is for true .
A expressão is será true se expr não for null e qualquer um dos seguintes for verdadeiro:
expr for uma instância do mesmo tipo que type.
expr for uma instância de um tipo derivado de type. Em outras palavras, o resultado de expr pode sofrer
upcast para uma instância de type.
expr tem um tipo de tempo de compilação que é uma classe base de type, e expr tem um tipo de tempo
de execução que é type ou é derivado de type. O tipo de tempo de compilação de uma variável é o tipo
da variável, conforme definido em sua declaração. O tipo de tempo de execução de uma variável é o tipo
da instância atribuída a essa variável.
expr é uma instância de um tipo que implementa a interface de type.
A partir do C# 7.1, expr pode ter um tipo de tempo de compilação definido por um parâmetro de tipo genérico
e suas restrições.
Se expr for true e is for usado com uma instrução if , varname será atribuído somente dentro da instrução
if . O escopo de varname é da expressão is até o final do bloco que envolve a instrução if . Usar varname
em qualquer outro local gera um erro em tempo de compilação para o uso de uma variável que não foi
atribuída.
O exemplo a seguir usa o padrão de tipo is para fornecer a implementação do método
IComparable.CompareTo(Object) de um tipo.
using System;
Sem a correspondência de padrões, esse código pode ser escrito da seguinte maneira. O uso da
correspondência de padrões de tipo produz código mais compacto e legível eliminando a necessidade de testar
se o resultado de uma conversão é um null .
using System;
O padrão de tipo is também produz código mais compacto ao determinar o tipo de um tipo de valor. O
exemplo a seguir usa o padrão de tipo is para determinar se um objeto é uma instância de Person ou Dog
antes de exibir o valor de uma propriedade adequada.
using System;
O código equivalente sem correspondência de padrões requer uma atribuição separada que inclui uma
conversão explícita.
using System;
Padrão de constante
Ao executar a correspondência de padrões com o padrão constante, is testa se uma expressão é igual a uma
constante especificada. No C# 6 e em versões anteriores, o padrão constante tem suporte da instrução switch. A
partir do C# 7.0, ele também é compatível com a instrução is . A sintaxe é:
expr is constant
em que expr é a expressão a ser avaliada e constant é o valor a ser testado. constant pode ser qualquer uma das
expressões de constante a seguir:
Um valor literal.
O nome de uma variável const declarada.
Uma constante de enumeração.
A expressão de constante é avaliada da seguinte forma:
Se expr e constant forem tipos integrais, o operador de igualdade de C# determinará se a expressão
retorna true (ou seja, se expr == constant ).
Caso contrário, o valor da expressão será determinado por uma chamada ao método estático
Object.Equals(expr, constant).
O exemplo a seguir combina os padrões de tipo e constante para testar se um objeto é uma instância de Dice
e, caso seja, para determinar se o valor de uma distribuição de dados é 6.
using System;
}
public int Roll()
{
return rnd.Next(1, 7);
}
}
class Program
{
static void Main(string[] args)
{
var d1 = new Dice();
ShowValue(d1);
}
É possível verificar em busca de null usando o padrão de constante. A palavra-chave null é compatível com
a instrução is . A sintaxe é:
expr is null
class Program
{
static void Main(string[] args)
{
object o = null;
if (o is null)
{
Console.WriteLine("o does not have a value");
}
else
{
Console.WriteLine($"o is {o}");
}
int? x = 10;
if (x is null)
{
Console.WriteLine("x does not have a value");
}
else
{
Console.WriteLine($"x is {x.Value}");
}
Padrão var
O padrão var é um catch-all para qualquer tipo ou valor. O valor de expr é sempre atribuído a uma variável
local do mesmo tipo que o tipo de tempo de compilação de expr. O resultado da expressão is é sempre true .
A sintaxe é:
O exemplo a seguir usa o padrão var para atribuir uma expressão a uma variável chamada obj . Em seguida,
ele exibe o valor e o tipo de obj .
using System;
class Program
{
static void Main()
{
object[] items = { new Book("The Tempest"), new Person("John") };
foreach (var item in items) {
if (item is var obj)
Console.WriteLine($"Type: {obj.GetType().Name}, Value: {obj}");
}
}
}
class Book
{
public Book(string title)
{
Title = title;
}
class Person
{
public Person(string name)
{
Name = name;
}
Especificação da linguagem C#
Para saber mais, confira a seção O operador is da especificação da linguagem C# e as seguintes propostas da
linguagem C#:
Correspondência de padrões
Correspondência de padrões com genéricos
Consulte também
Referência de C#
Palavras-chave do C#
Operadores cast e teste de tipo
Restrição new (Referência em C#)
23/10/2019 • 2 minutes to read • Edit Online
A restrição new especifica que um argumento de tipo em uma declaração de classe genérica deve ter um
construtor público sem parâmetros. Para usar a restrição new , o tipo não pode ser abstrato.
Aplique a restrição new a um parâmetro de tipo quando uma classe genérica criar novas instâncias do tipo,
conforme mostrado no exemplo a seguir:
Quando você usa a restrição new() com outras restrições, ela deve ser especificada por último:
Especificação da linguagem C#
Para obter mais informações, confira a seção Restrições de parâmetro de tipo na Especificação da linguagem C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Genéricos
where (restrição de tipo genérico) (Referência de C#)
25/11/2019 • 7 minutes to read • Edit Online
A cláusula where em uma definição genérica especifica restrições sobre os tipos que são usados como
argumentos para parâmetros de tipo em um tipo genérico, método, delegado ou função local. As restrições
podem especificar interfaces, classes base ou exigir que um tipo genérico seja uma referência, um valor ou um
tipo não gerenciado. Elas declaram funcionalidades que o argumento de tipo deve ter.
Por exemplo, você pode declarar uma classe genérica, MyGenericClass , de modo que o parâmetro de tipo T
implementa a interface IComparable<T>:
NOTE
Para obter mais informações sobre a cláusula where em uma expressão de consulta, consulte Cláusula where.
A cláusula where também pode incluir uma restrição de classe base. A restrição de classe base declara que um
tipo a ser usado como um argumento de tipo para aquele tipo genérico tem a classe especificada como uma
classe base (ou é que a classe base) a ser usada como um argumento de tipo para aquele tipo genérico. Se a
restrição de classe base for usada, ela deverá aparecer antes de qualquer outra restrição nesse parâmetro de tipo.
Alguns tipos não têm permissão como uma restrição de classe base: Object, Array e ValueType. Antes do C# 7.3,
Enum, Delegate e MulticastDelegate também não tinham permissão como restrições de classe base. O exemplo a
seguir mostra os tipos que agora podem ser especificados como classe base:
A cláusula where pode especificar que o tipo é um class ou um struct . A restrição struct elimina a
necessidade de especificar uma restrição de classe base de System.ValueType . O tipo System.ValueType não pode
ser usado como uma restrição de classe base. O exemplo a seguir mostra as restrições class e struct :
A cláusula where pode incluir a restrição notnull . A restrição notnull limita o parâmetro de tipo a tipos não
anuláveis. Esse tipo pode ser um tipo de valor ou um tipo de referência não anulável. A restrição notnull está
disponível a partir C# de 8,0 para o código compilado em um contexto de nullable enable . Ao contrário de outras
restrições, se um argumento de tipo violar a restrição de notnull , o compilador gerará um aviso em vez de um
erro. Os avisos são gerados apenas em um contexto de nullable enable .
IMPORTANT
Declarações genéricas que incluem a restrição notnull podem ser usadas em um contexto alheios anulável, mas o
compilador não impõe a restrição.
#nullable enable
class NotNullContainer<T>
where T : notnull
{
}
#nullable restore
A cláusula where também pode incluir uma restrição unmanaged . A restrição unmanaged limita o parâmetro de
tipo a tipos conhecidos como tipos não gerenciados. Usando a restrição unmanaged , é mais fácil escrever o código
de interoperabilidade de nível baixo em C#. Essa restrição habilita rotinas reutilizáveis em todos os tipos não
gerenciados. A restrição unmanaged não pode ser combinada à restrição class ou struct . A restrição unmanaged
impõe que o tipo deve ser um struct :
class UnManagedWrapper<T>
where T : unmanaged
{ }
A cláusula where também pode incluir uma restrição de construtor, new() . Essa restrição torna possível criar
uma instância de um parâmetro de tipo usando o operador new . A restrição New () permite que o compilador
saiba que qualquer argumento de tipo fornecido deve ter um construtor sem parâmetros acessível. Por exemplo:
A restrição new() aparece por último na cláusula where . A restrição new() não pode ser combinada às
restrições struct ou unmanaged . Todos os tipos que satisfazem as restrições devem ter um construtor sem
parâmetros acessível, tornando a restrição new() redundante.
Com vários parâmetros de tipo, use uma cláusula where para cada parâmetro de tipo, por exemplo:
namespace CodeExample
{
class Dictionary<TKey, TVal>
where TKey : IComparable<TKey>
where TVal : IMyInterface
{
public void Add(TKey key, TVal val) { }
}
}
Você também pode anexar restrições a parâmetros de tipo de métodos genéricos, como mostrado no exemplo a
seguir:
public void MyMethod<T>(T t) where T : IMyInterface { }
Observe que a sintaxe para descrever as restrições de parâmetro de tipo em delegados é a mesma que a dos
métodos:
Especificação da linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a fonte
definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
Introdução aos genéricos
Restrição new
Restrições a parâmetros de tipo
base (Referência de C#)
23/10/2019 • 3 minutes to read • Edit Online
A palavra-chave base é usada para acessar membros de classe base de dentro de uma classe derivada:
Chamar um método que foi substituído por outro método na classe base.
Especificar qual construtor de classe base deve ser chamado ao criar instâncias da classe derivada.
Um acesso de classe base é permitido somente em um construtor, em um método de instância ou em um
acessador de propriedade de instância.
É um erro usar a palavra-chave base de dentro de um método estático.
A classe base que é acessada é a que é especificada na declaração de classe. Por exemplo, se você especificar
class ClassB : ClassA , os membros da ClassA são acessados da ClassB, independentemente da classe base
da ClassA.
Exemplo
Neste exemplo, as classes base Person e a classe derivada Employee têm um método chamado Getinfo .
Usando a palavra-chave base , é possível chamar o método Getinfo na classe base, de dentro da classe
derivada.
public class Person
{
protected string ssn = "444-55-6666";
protected string name = "John L. Malgraine";
class TestClass
{
static void Main()
{
Employee E = new Employee();
E.GetInfo();
}
}
/*
Output
Name: John L. Malgraine
SSN: 444-55-6666
Employee ID: ABC567EFG
*/
Exemplo
Este exemplo mostra como especificar o construtor da classe base chamado ao criar instâncias de uma classe
derivada.
public class BaseClass
{
int num;
public BaseClass()
{
Console.WriteLine("in BaseClass()");
}
public BaseClass(int i)
{
num = i;
Console.WriteLine("in BaseClass(int i)");
}
Especificação da linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a
fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
this
this (Referência de C#)
23/10/2019 • 2 minutes to read • Edit Online
A palavra-chave this refere-se à instância atual da classe e também é usada como um modificador do
primeiro parâmetro de um método de extensão.
NOTE
Este artigo discute o uso de this com instâncias de classe. Para obter mais informações sobre seu uso em métodos de
extensão, consulte Métodos de extensão.
Para passar um objeto como parâmetro para outros métodos, por exemplo:
CalcTax(this);
Funções de membro estático, por existirem no nível da classe e não como parte de um objeto, não têm um
ponteiro this . É um erro se referir a this em um método estático.
Exemplo
Neste exemplo, this é usado para qualificar os membros de classe Employee , name e alias , que são
ocultados por nomes semelhantes. Ele também é usado para passar um objeto para o método CalcTax , que
pertence a outra classe.
class Employee
{
private string name;
private string alias;
private decimal salary = 3000.00m;
// Constructor:
public Employee(string name, string alias)
{
// Use this to qualify the fields, name and alias:
this.name = name;
this.alias = alias;
}
// Printing method:
public void printEmployee()
{
Console.WriteLine("Name: {0}\nAlias: {1}", name, alias);
// Passing the object to the CalcTax method by using this:
Console.WriteLine("Taxes: {0:C}", Tax.CalcTax(this));
}
class Tax
{
public static decimal CalcTax(Employee E)
{
return 0.08m * E.Salary;
}
}
class MainClass
{
static void Main()
{
// Create objects:
Employee E1 = new Employee("Mingda Pan", "mpan");
// Display results:
E1.printEmployee();
}
}
/*
Output:
Name: Mingda Pan
Alias: mpan
Taxes: $240.00
*/
Especificação da linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a
fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
base
Métodos
null (Referência de C#)
08/11/2019 • 2 minutes to read • Edit Online
A palavra-chave null é um literal que representa uma referência nula, que não faz referência a qualquer
objeto. null é o valor padrão de variáveis do tipo de referência. Tipos de valor comum não podem ser
nulos, exceto para tipos de valores anuláveis.
O exemplo a seguir demonstra alguns comportamentos da palavra-chave null:
class Program
{
class MyClass
{
public void MyMethod() { }
}
// Returns true.
Console.WriteLine("null == null is {0}", null == null);
}
}
Especificação da linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a
fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Palavras-chave do C#
Tabela de valores padrão
Nada (Visual Basic)
2 minutes to read
2 minutes to read
default (Referência de C#)
23/10/2019 • 2 minutes to read • Edit Online
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Palavras-chave contextuais (Referência de C#)
04/11/2019 • 2 minutes to read • Edit Online
Uma palavra-chave contextual é usada para fornecer um significado específico no código, mas não é uma palavra
reservada no C#. As seguintes palavras-chave contextuais são apresentadas nesta seção:
PALAVRA-CHAVE DESCRIÇÃO
Todas as palavras-chave de consulta introduzidas no C# 3.0 também são contextuais. Para obter mais informações,
consulte Palavras-chave de consulta (LINQ ).
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
add (Referência de C#)
23/10/2019 • 2 minutes to read • Edit Online
A palavra-chave contextual add é usada para definir um acessador de evento personalizado que é invocado
quando o código cliente assina seu evento. Se você fornecer um acessador add personalizado, também será
fornecer um acessador remove.
Exemplo
O exemplo a seguir mostra um evento que tem acessadores add e remove personalizados. Para ver o exemplo
completo, confira Como implementar eventos de interface.
Normalmente, não é necessário fornecer seus próprios acessadores de eventos personalizados. Os acessadores
que são gerados automaticamente pelo compilador quando você declara um evento são suficientes para a
maioria dos cenários.
Consulte também
Eventos
get (Referência de C#)
23/10/2019 • 2 minutes to read • Edit Online
A palavra-chave get define um método do acessador em uma propriedade ou um indexador que retorna
o valor da propriedade ou o elemento do indexador. Para obter mais informações, consulte Propriedades,
Propriedades autoimplementadas e Indexadores.
O exemplo a seguir define um acessador get e um acessador set para uma propriedade chamada
Seconds . Ela usa um campo particular chamado _seconds para dar suporte ao valor da propriedade.
class TimePeriod
{
private double _seconds;
Geralmente, o acessador get consiste em uma única instrução que retorna um valor, como no exemplo
anterior. Começando com o C# 7.0, você pode implementar o acessador get como um membro apto para
expressão. O exemplo a seguir implementa os acessadores get e set como membros aptos para
expressão.
class TimePeriod
{
private double _seconds;
Para casos simples em que os acessadores get e set de uma propriedade não realizam nenhuma outra
operação, a não ser a configuração ou a recuperação de um valor em um campo de suporte particular, você
pode tirar proveito do suporte do compilador do C# para propriedades autoimplementadas. O exemplo a
seguir implementa Hours como uma propriedade autoimplementada.
class TimePeriod2
{
public double Hours { get; set; }
}
Especificação da Linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a
fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Propriedades
tipo parcial (Referência em C#)
04/11/2019 • 2 minutes to read • Edit Online
Definições de tipo parcial permitem que a definição de uma classe, struct ou interface seja dividida em vários
arquivos.
Em File1.cs:
namespace PC
{
partial class A
{
int num = 0;
void MethodA() { }
partial void MethodC();
}
}
Em File2.cs, a declaração:
namespace PC
{
partial class A
{
void MethodB() { }
partial void MethodC() { }
}
}
Comentários
Dividir um tipo de classe, struct ou interface em vários arquivos pode ser útil quando você está trabalhando com
projetos grandes ou com o código gerado automaticamente, como o código fornecido pelo Designer de
Formulários do Windows. Um tipo parcial pode conter um método parcial. Para obter mais informações,
consulte Classes parciais e métodos.
Especificação da linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a fonte
definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
Modificadores
Introdução aos genéricos
Método parcial (C# Reference)
23/10/2019 • 2 minutes to read • Edit Online
Um método parcial tem sua assinatura definida em uma parte de um tipo parcial e sua implementação definida
em outra parte do tipo. Os métodos parciais permitem que os designers de classe forneçam ganchos de método,
semelhantes a manipuladores de eventos, que os desenvolvedores podem decidir implementar ou não. Se o
desenvolvedor não fornecer uma implementação, o compilador removerá a assinatura no tempo de compilação.
As seguintes condições são aplicáveis a métodos parciais:
As assinaturas em ambas as partes do tipo parcial devem ser correspondentes.
O método deve retornar nulo.
Não é permitido nenhum modificador de acesso. Métodos parciais são implicitamente privados.
O exemplo a seguir mostra um método parcial definido em duas partes de uma classe parcial:
namespace PM
{
partial class A
{
partial void OnSomethingHappened(string s);
}
Consulte também
Referência de C#
Tipo parcial
remove (Referência de C#)
23/10/2019 • 2 minutes to read • Edit Online
A palavra-chave contextual remove é usada para definir um acessador de eventos personalizado invocado
quando o código cliente cancela a assinatura do seu evento. Se você fornecer um acessador remove
personalizado, também será necessário fornecer um acessador add.
Exemplo
O exemplo a seguir mostra um evento com os acessadores add e remove personalizados. Para ver o exemplo
completo, confira Como implementar eventos de interface.
Normalmente, não é necessário fornecer seus próprios acessadores de eventos personalizados. Os acessadores
que são gerados automaticamente pelo compilador quando você declara um evento são suficientes para a
maioria dos cenários.
Consulte também
Eventos
set (Referência de C#)
23/10/2019 • 2 minutes to read • Edit Online
A palavra-chave set define um método acessador em uma propriedade ou indexador que atribui um valor
ao elemento da propriedade ou do elemento. Para obter mais informações e exemplos, consulte
Propriedades, Propriedades autoimplementadas e Indexadores.
O exemplo a seguir define um acessador get e um acessador set para uma propriedade chamada
Seconds . Ela usa um campo particular chamado _seconds para dar suporte ao valor da propriedade.
class TimePeriod
{
private double _seconds;
Geralmente, o acessador set consiste em uma única instrução que retorna um valor, como no exemplo
anterior. Começando com o C# 7.0, você pode implementar o acessador set como um membro apto para
expressão. O exemplo a seguir implementa os acessadores get e set como membros com corpo de
expressão.
class TimePeriod
{
private double _seconds;
Para casos simples em que os acessadores get e set de uma propriedade não realizam nenhuma outra
operação, a não ser a configuração ou a recuperação de um valor em um campo de suporte particular, você
pode tirar proveito do suporte do compilador do C# para propriedades autoimplementadas. O exemplo a
seguir implementa Hours como uma propriedade autoimplementada.
class TimePeriod2
{
public double Hours { get; set; }
}
Especificação da linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a
fonte definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
Propriedades
when (Referência de C#)
23/10/2019 • 4 minutes to read • Edit Online
Você pode usar a palavra-chave contextual when para especificar uma condição de filtro em dois contextos:
Na instrução catch de um bloco try/catch ou try/catch/finally.
No rótulo case de uma instrução switch.
em que expr é uma expressão que é avaliada como um valor booliano. Se ele retornar true , o manipulador de
exceção será executado, se false , não executará.
O exemplo a seguir usa a palavra-chave when para executar os manipuladores condicionalmente para um
HttpRequestException dependendo do texto da mensagem de exceção.
using System;
using System.Net.Http;
using System.Threading.Tasks;
class Program
{
static void Main()
{
Console.WriteLine(MakeRequest().Result);
}
em que expr é um padrão de constante ou padrão de tipo que é comparado com a expressão de correspondência e
when-condition é qualquer expressão booliana.
O exemplo a seguir usa a palavra-chave when para testar os objetos Shape que têm uma área de zero, bem como
para testar uma variedade de objetos Shape que têm uma área maior que zero.
using System;
Consulte também
instrução switch
instruções try/catch
instrução try/catch/finally
value (Referência de C#)
24/10/2019 • 2 minutes to read • Edit Online
class MyBaseClass
{
// virtual auto-implemented property. Overrides can only
// provide specialized behavior if they implement get and set accessors.
public virtual string Name { get; set; }
Especificação da linguagem C#
Para obter mais informações, consulte a Especificação da linguagem C#. A especificação da linguagem é a fonte
definitiva para a sintaxe e o uso de C#.
Consulte também
Referência de C#
Guia de Programação em C#
Palavras-chave do C#
yield (Referência de C#)
23/10/2019 • 7 minutes to read • Edit Online
Ao usar a palavra-chave contextual yield em uma instrução, você indica que o método, o operador ou o
acessador get em que ela é exibida é um iterador. Usar yield para definir um iterador elimina a
necessidade de uma classe adicional explícita (a classe que mantém o estado de uma enumeração,
consulte IEnumerator<T> para obter um exemplo) ao implementar o padrão IEnumerable e IEnumerator
para um tipo de coleção personalizado.
O exemplo a seguir mostra as duas formas de instrução yield .
Comentários
Você usa uma instrução yield return para retornar cada elemento individualmente.
A sequência retornada de um método iterador pode ser consumida usando uma instrução foreach ou
uma consulta LINQ. Cada iteração do loop foreach chama o método iterador. Quando uma instrução
yield return é atingida no método iterador, expression é retornado e o local atual no código é retido. A
execução será reiniciada desse local na próxima vez que a função iteradora for chamada.
Você pode usar uma instrução yield break para terminar a iteração.
Para obter mais informações sobre iteradores, consulte Iteradores.
Tratamento de exceções
Uma instrução yield return não pode estar localizada em um bloco try-catch. Uma instrução
yield return pode estar localizada no bloco try de uma instrução try-finally.
Uma instrução yield break pode estar localizada em um bloco try ou em um bloco catch, mas não em
um bloco finally.
Se o corpo foreach (fora do método iterador) acionar uma exceção, um bloco finally no método
iterador será executado.
Implementação técnica
O código a seguir retorna uma IEnumerable<string> de um método iterador e itera através de seus
elementos.
A chamada a MyIteratorMethod não executa o corpo do método. Em vez disso, a chamada retorna
IEnumerable<string> na variável elements .
Em uma iteração do loop foreach , o método MoveNext é chamado para elements . Essa chamada
executará o corpo de MyIteratorMethod até que a próxima instrução yield return seja atingida. A
expressão retornada pela instrução yield return determina não apenas o valor da variável element para
o consumo do corpo do loop, mas também a propriedade Current de elements , que é uma
IEnumerable<string> .
Em cada iteração subsequente do loop foreach , a execução do corpo do iterador continuará de onde
parou, parando novamente quando atingir uma instrução yield return . O loop foreach é concluído
quando o fim do método iterador ou uma instrução yield break é atingida.
Exemplo
O exemplo a seguir contém uma instrução yield return dentro de um loop for . Cada iteração do corpo
da instrução foreach no método Main cria uma chamada à função iteradora Power . Cada chamada à
função iteradora prossegue para a próxima execução da instrução yield return que ocorre durante a
próxima iteração do loop for .
O tipo de retorno do método iterador é IEnumerable que é um tipo de interface de iterador. Quando o
método iterador é chamado, ele retorna um objeto enumerável que contém as potências de um número.
public class PowersOf2
{
static void Main()
{
// Display powers of 2 up to the exponent of 8:
foreach (int i in Power(2, 8))
{
Console.Write("{0} ", i);
}
}
Exemplo
O exemplo a seguir demonstra um acessador get que é um iterador. No exemplo, cada instrução
yield return retorna uma instância de uma classe definida pelo usuário.
Consulte também
Referência de C#
Guia de Programação em C#
foreach, in
Iteradores
Palavras-chave de consulta (Referência de C#)
04/11/2019 • 2 minutes to read • Edit Online
Nesta seção
CLÁUSULA DESCRIÇÃO
Consulte também
Palavras-chave do C#
LINQ (Consulta Integrada à Linguagem)
LINQ em C#
Introdução a LINQ em C#
Cláusula from (Referência de C#)
23/10/2019 • 8 minutes to read • Edit Online
Uma expressão de consulta deve começar com uma cláusula from . Além disso, uma expressão de consulta
pode conter subconsultas, que também começam com uma cláusula from . A cláusula from especifica o
seguinte:
A fonte de dados na qual a consulta ou subconsulta será executada.
Uma variável de intervalo local que representa cada elemento na sequência de origem.
A variável de intervalo e a fonte de dados são fortemente tipadas. A fonte de dados referenciada na cláusula
from deve ter um tipo de IEnumerable, IEnumerable<T> ou um tipo derivado, por exemplo, IQueryable<T>.
No exemplo a seguir, numbers é a fonte de dados e num é a variável de intervalo. Observe que ambas as
variáveis são fortemente tipadas, mesmo com o uso da palavra-chave var.
class LowNums
{
static void Main()
{
// A simple data source.
int[] numbers = { 5, 4, 1, 3, 9, 8, 6, 7, 2, 0 };
A variável de intervalo
O compilador infere que o tipo da variável de intervalo quando a fonte de dados implementa
IEnumerable<T>. Por exemplo, se a fonte tem um tipo de IEnumerable<Customer> , então, a variável de
intervalo será inferida como Customer . O tipo deve ser especificado explicitamente somente quando a fonte
for um tipo IEnumerable não genérico, como ArrayList. Para obter mais informações, confira Como:
Consultar um ArrayList com LINQ.
No exemplo anterior, num é inferido como do tipo int . Como a variável de intervalo é fortemente tipada, é
possível chamar métodos nela ou usá-la em outras operações. Por exemplo, em vez de gravar select num ,
grave select num.ToString() para fazer com que a expressão de consulta retorne uma sequência de cadeias
de caracteres em vez de números inteiros. Também é possível gravar select num + 10 para fazer com que a
expressão retorne a sequência 14, 11, 13, 12, 10. Para obter mais informações, consulte cláusula select.
A variável de intervalo é como uma variável de iteração em uma instrução foreach, com a exceção de uma
diferença muito importante: na verdade, uma variável de intervalo nunca armazena dados da fonte. É apenas
uma conveniência sintática que habilita a consulta a descrever o que ocorrerá quando ela for executada. Para
obter mais informações, consulte Introdução a Consultas de LINQ (C#).
// Use a compound from to access the inner sequence within each element.
// Note the similarity to a nested foreach statement.
var scoreQuery = from student in students
from score in student.Scores
where score > 90
select new { Last = student.LastName, score };
class CompoundFrom2
{
static void Main()
{
char[] upperCase = { 'A', 'B', 'C' };
char[] lowerCase = { 'x', 'y', 'z' };
Console.WriteLine("Filtered non-equijoin:");
// Rest the mouse pointer over joinQuery2 to verify its type.
foreach (var pair in joinQuery2)
{
Console.WriteLine("{0} is matched to {1}", pair.lower, pair.upper);
}
Consulte também
Palavras-chave de Consulta (LINQ )
LINQ (Consulta Integrada à Linguagem)
Cláusula where (Referência de C#)
04/11/2019 • 4 minutes to read • Edit Online
A cláusula where é usada em uma expressão de consulta para especificar quais elementos da fonte de dados
serão retornados na expressão de consulta. Aplica-se uma condição booliana (predicate) para cada elemento
de origem (referenciado pela variável de intervalo) e retorna aqueles para os quais a condição especificada
for verdadeira. Uma única expressão de consulta pode conter várias cláusulas where e uma única cláusula
pode conter várias subexpressões de predicado.
Exemplo
No exemplo a seguir, a cláusula where filtra todos os números, exceto aqueles que são menores que cinco.
Se você remover a cláusula where , todos os números da fonte de dados serão retornados. A expressão
num < 5 é o predicado aplicado a cada elemento.
class WhereSample
{
static void Main()
{
// Simple data source. Arrays support IEnumerable<T>.
int[] numbers = { 5, 4, 1, 3, 9, 8, 6, 7, 2, 0 };
Exemplo
Dentro de uma única cláusula where , você pode especificar tantos predicados quanto necessário usando os
operadores && e ||. No exemplo a seguir, a consulta especifica dois predicados para selecionar apenas os
números pares que são menores que cinco.
class WhereSample2
{
static void Main()
{
// Data source.
int[] numbers = { 5, 4, 1, 3, 9, 8, 6, 7, 2, 0 };
}
}
// Output:
// 4 2 0
// 4 2 0
Exemplo
Uma cláusula where pode conter um ou mais métodos que retornam valores boolianos. No exemplo a
seguir, a cláusula where usa um método para determinar se o valor atual da variável de intervalo é par ou
ímpar.
class WhereSample3
{
static void Main()
{
// Data source
int[] numbers = { 5, 4, 1, 3, 9, 8, 6, 7, 2, 0 };
Comentários
A cláusula where é um mecanismo de filtragem. Ela pode ser posicionada em quase qualquer lugar em uma
expressão de consulta, exceto que ela não pode ser a primeira ou a última cláusula. A cláusula where pode
aparecer antes ou depois de uma cláusula group dependendo se você tiver que filtrar os elementos de
origem antes ou depois de eles serem agrupados.
Se um predicado especificado não for válido para os elementos na fonte de dados, o resultado será um erro
em tempo de compilação. Essa é uma vantagem da verificação de tipo forte fornecida pelo LINQ.
Em tempo de compilação, a palavra-chave where é convertida em uma chamada para o método de
operador de consulta padrão Where.
Consulte também
Palavras-chave de Consulta (LINQ )
Cláusula From
Cláusula select
Filtrando Dados
LINQ em C#
Introdução a LINQ em C#
Cláusula select (Referência de C#)
04/11/2019 • 7 minutes to read • Edit Online
Em uma expressão de consulta, a cláusula select especifica o tipo de valores que serão produzidos quando
a consulta é executada. O resultado é baseado na avaliação de todas as cláusulas anteriores e em quaisquer
expressões na cláusula select em si. Uma expressão de consulta deve terminar com uma cláusula select
ou uma cláusula group.
O exemplo a seguir mostra uma cláusula select simples em uma expressão de consulta.
class SelectSample1
{
static void Main()
{
//Create the data source
List<int> Scores = new List<int>() { 97, 92, 81, 60 };
O tipo da sequência produzida pela cláusula select determina o tipo da variável de consulta
queryHighScores . No caso mais simples, a cláusula select apenas especifica a variável de intervalo. Isso faz
com que a sequência retornada contenha elementos do mesmo tipo que a fonte de dados. Para obter mais
informações, consulte Relacionamentos de tipo em operações de consulta LINQ. No entanto, a cláusula
select também fornece um mecanismo poderoso para transformar (ou projetar) dados de origem em
novos tipos. Para obter mais informações, consulte Transformações de dados com LINQ (C#).
Exemplo
O exemplo a seguir mostra todas as diferentes formas que uma cláusula select pode tomar. Em cada
consulta, observe a relação entre a cláusula select e o tipo da variável de consulta ( studentQuery1 ,
studentQuery2 e assim por diante).
class SelectSample2
{
// Define some classes
public class Student
{
public string First { get; set; }
public string Last { get; set; }
public int ID { get; set; }
public List<int> Scores;
public ContactInfo GetContactInfo(SelectSample2 app, int id)
{
{
ContactInfo cInfo =
(from ci in app.contactList
where ci.ID == id
select ci)
.FirstOrDefault();
return cInfo;
}
Comentários
No tempo de compilação, a cláusula select é convertida em uma chamada de método para o operador de
consulta padrão Select.
Consulte também
Referência de C#
Palavras-chave de Consulta (LINQ )
Cláusula From
partial (método) (Referência do C#)
Tipos Anônimos
LINQ em C#
Introdução a LINQ em C#
Cláusula group (Referência de C#)
27/11/2019 • 12 minutes to read • Edit Online
A cláusula group retorna uma sequência de objetos IGrouping<TKey,TElement> que contêm zero ou mais
itens que correspondem ao valor de chave do grupo. Por exemplo, é possível agrupar uma sequência de
cadeias de caracteres de acordo com a primeira letra de cada cadeia de caracteres. Nesse caso, a primeira
letra é a chave, tem um tipo char e é armazenada na propriedade Key de cada objeto
IGrouping<TKey,TElement>. O compilador infere o tipo da chave.
É possível finalizar uma expressão de consulta com uma cláusula group , conforme mostrado no exemplo a
seguir:
Caso deseje executar mais operações de consulta em cada grupo, é possível especificar um identificador
temporário usando a palavra-chave contextual into. Ao usar into , é necessário continuar a consulta e, em
algum momento, finalizá-la com uma instrução select ou outra cláusula group , conforme mostrado no
trecho a seguir:
Exemplos mais completos do uso de group com e sem into serão apresentados na seção Exemplo deste
artigo.
Tipos de chave
As chaves de grupo podem ser de qualquer tipo, como uma cadeia de caracteres, um tipo numérico
interno, um tipo nomeado definido pelo usuário ou um tipo anônimo.
Agrupar por cadeia de caracteres
Os exemplos de código anteriores usaram um char . Em vez disso, uma chave de cadeia de caracteres
pode facilmente ter sido especificada, por exemplo, o sobrenome completo:
// Same as previous example except we use the entire last name as a key.
// Query variable is an IEnumerable<IGrouping<string, Student>>
var studentQuery3 =
from student in students
group student by student.Last;
class GroupSample1
{
// The element type of the data source.
public class Student
{
public string First { get; set; }
public string Last { get; set; }
public int ID { get; set; }
public List<int> Scores;
}
return students;
}
class GroupSample2
{
// The element type of the data source.
public class Student
{
public string First { get; set; }
public string Last { get; set; }
public int ID { get; set; }
public List<int> Scores;
}
return students;
Use um tipo nomeado se for necessário passar a variável de consulta para outro método. Crie uma classe
especial usando as propriedades autoimplementadas das chaves e, em seguida, substitua os métodos
Equals e GetHashCode. Também é possível usar um struct; nesse caso, não é exatamente necessário
substituir esses métodos. Para obter mais informações, consulte como implementar uma classe leve com
propriedades implementadas automaticamente e como consultar arquivos duplicados em uma árvore de
diretórios. O último artigo apresenta um exemplo de código que demonstra como usar uma chave
composta com um tipo nomeado.
Exemplo
O exemplo a seguir mostra a norma padrão para ordenar dados de origem em grupos quando nenhuma
lógica de consulta adicional for aplicada aos grupos. Isso é chamado de “agrupamento sem uma
continuação”. Os elementos em uma matriz de cadeias de caracteres são agrupados de acordo com a
primeira letra. O resultado da consulta é um tipo IGrouping<TKey,TElement> que contém uma
propriedade Key pública do tipo char e uma coleção IEnumerable<T> que contém cada item no
agrupamento.
O resultado de uma cláusula group é uma sequência de sequências. Portanto, para acessar os elementos
individuais dentro de cada grupo retornado, use um loop aninhado foreach dentro do loop que itera as
chaves de grupo, conforme mostrado no exemplo a seguir.
class GroupExample1
{
static void Main()
{
// Create a data source.
string[] words = { "blueberry", "chimpanzee", "abacus", "banana", "apple", "cheese" };
Exemplo
Este exemplo mostra como executar a lógica adicional nos grupos depois criá-los, usando uma continuação
com into . Para obter mais informações, consulte into. O exemplo a seguir consulta cada grupo para
selecionar apenas aqueles cujo valor da chave é uma vogal.
class GroupClauseExample2
{
static void Main()
{
// Create the data source.
string[] words2 = { "blueberry", "chimpanzee", "abacus", "banana", "apple", "cheese",
"elephant", "umbrella", "anteater" };
Comentários
No tempo de compilação, as cláusulas group são convertidas em chamadas para o método GroupBy.
Consulte também
IGrouping<TKey,TElement>
GroupBy
ThenBy
ThenByDescending
Palavras-chave de consulta
LINQ (Consulta Integrada à Linguagem)
Criar um grupo aninhado
Agrupar resultados de consultas
Executar uma subconsulta em uma operação de agrupamento
into (Referência de C#)
04/11/2019 • 2 minutes to read • Edit Online
A palavra-chave contextual into pode ser usada para criar um identificador temporário para armazenar os
resultados de uma cláusula group, join ou select em um novo identificador. Esse identificador por si só pode ser
um gerador de comandos de consulta adicionais. Quando usado em uma cláusula group ou select , o uso do
novo identificador é, às vezes, conhecido como uma continuação.
Exemplo
O exemplo a seguir mostra o uso da palavra-chave into para habilitar um identificador temporário
fruitGroup que tem um tipo inferido de IGrouping . Usando o identificador, é possível invocar o método Count
em cada grupo e selecionar apenas os grupos que contêm duas ou mais palavras.
class IntoSample1
{
static void Main()
{
// Execute the query. Note that we only iterate over the groups,
// not the items in each group
foreach (var item in wordGroups1)
{
Console.WriteLine(" {0} has {1} elements.", item.FirstLetter, item.Words);
}
O uso de into em uma cláusula group só é necessário quando você deseja realizar operações de consulta
adicionais em cada grupo. Para obter mais informações, consulte Cláusula group.
Para obter um exemplo do uso de into em uma cláusula join , consulte cláusula join.
Consulte também
Palavras-chave de Consulta (LINQ )
LINQ em C#
Cláusula group
Cláusula orderby (Referência de C#)
04/11/2019 • 3 minutes to read • Edit Online
Em uma expressão de consulta, a cláusula orderby faz com que a sequência ou subsequência (grupo)
retornada seja classificada em ordem crescente ou decrescente. Várias chaves podem ser especificadas para
executar uma ou mais operações de classificação secundárias. A classificação é executada pelo comparador
padrão para o tipo do elemento. A ordem de classificação crescente é padrão. Também é possível
especificar um comparador personalizado. No entanto, está disponível somente por meio da sintaxe
baseada em método. Para obter mais informações, consulte Classificando dados.
Exemplo
No exemplo a seguir, a primeira consulta classifica as palavras em ordem alfabética começando em A e a
segunda consulta classifica as mesmas palavras em ordem decrescente. (A palavra-chave ascending é o
valor de classificação padrão e pode ser omitida.)
class OrderbySample1
{
static void Main()
{
// Create a delicious data source.
string[] fruits = { "cherry", "apple", "blueberry" };
Descending:
cherry
blueberry
apple
*/
Exemplo
O exemplo a seguir executa uma classificação primária pelos sobrenomes dos alunos e, em seguida, uma
classificação secundária pelos seus nomes.
class OrderbySample2
{
// The element type of the data source.
public class Student
{
public string First { get; set; }
public string Last { get; set; }
public int ID { get; set; }
}
public static List<Student> GetStudents()
{
// Use a collection initializer to create the data source. Note that each element
// in the list contains an inner sequence of scores.
List<Student> students = new List<Student>
{
new Student {First="Svetlana", Last="Omelchenko", ID=111},
new Student {First="Claire", Last="O'Donnell", ID=112},
new Student {First="Sven", Last="Mortensen", ID=113},
new Student {First="Cesar", Last="Garcia", ID=114},
new Student {First="Debra", Last="Garcia", ID=115}
};
return students;
}
static void Main(string[] args)
{
// Create the data source.
List<Student> students = GetStudents();
// Now create groups and sort the groups. The query first sorts the names
// of all students so that they will be in alphabetical order after they are
// grouped. The second orderby sorts the group keys in alpha order.
var sortedGroups =
from student in students
orderby student.Last, student.First
group student by student.Last[0] into newGroup
orderby newGroup.Key
select newGroup;
sortedGroups:
G
Garcia, Cesar
Garcia, Cesar
Garcia, Debra
M
Mortensen, Sven
O
O'Donnell, Claire
Omelchenko, Svetlana
*/
Comentários
Em tempo de compilação, a cláusula orderby é convertida em uma chamada para o método OrderBy.
Várias chaves na cláusula orderby são traduzidas para chamadas de método ThenBy.
Consulte também
Referência de C#
Palavras-chave de Consulta (LINQ )
LINQ (Consulta Integrada à Linguagem)
Cláusula group
Introdução a LINQ em C#
Cláusula join (Referência de C#)
23/10/2019 • 15 minutes to read • Edit Online
A cláusula join é útil para associar elementos de sequências de origem diferentes que não têm
nenhuma relação direta no modelo de objeto. O único requisito é que os elementos em cada fonte
compartilhem algum valor que possa ser comparado pela igualdade. Por exemplo, um distribuidor de
alimentos pode ter uma lista de fornecedores de um determinado produto e uma lista de compradores.
Uma cláusula join pode ser usada, por exemplo, para criar uma lista de fornecedores e compradores
daquele produto, que estejam na mesma região especificada.
Uma cláusula join recebe duas sequências de origem como entrada. Os elementos em cada sequência
devem ser ou conter uma propriedade que possa ser comparada com uma propriedade correspondente
na outra sequência. A cláusula join compara a igualdade das chaves especificadas, usando a palavra-
chave especial equals . Todas as junções realizadas pela cláusula join são junções por igualdade. A
forma da saída de uma cláusula join depende do tipo específico de junção que você está realizando. A
seguir estão os três tipos de junção mais comuns:
Junção interna
Junção de grupo
Junção externa esquerda
Junção interna
O exemplo a seguir mostra uma junção por igualdade interna simples. Essa consulta produz uma
sequência simples de pares "nome de produto / categoria". A mesma cadeia de caracteres de categoria
aparecerá em vários elementos. Se um elemento de categories não tiver products correspondente,
essa categoria não aparecerá nos resultados.
var innerJoinQuery =
from category in categories
join prod in products on category.ID equals prod.CategoryID
select new { ProductName = prod.Name, Category = category.Name }; //produces flat sequence
Junção de grupo
Uma cláusula join com um expressão into é chamada de junção de grupo.
var innerGroupJoinQuery =
from category in categories
join prod in products on category.ID equals prod.CategoryID into prodGroup
select new { CategoryName = category.Name, Products = prodGroup };
Uma junção de grupo produz uma sequência de resultados hierárquicos, que associa os elementos na
sequência de origem à esquerda com um ou mais elementos correspondentes na sequência de origem
do lado direito. Uma junção de grupo não tem nenhum equivalente em termos relacionais. Ela é,
essencialmente, uma sequência de matrizes de objetos.
Se nenhum elemento da sequência de origem à direita que corresponda a um elemento na origem à
esquerda for encontrado, a cláusula join produzirá uma matriz vazia para aquele item. Portanto, a
junção de grupo é, basicamente, uma junção por igualdade interna, exceto pelo fato de que a sequência
de resultado é organizada em grupos.
É só selecionar os resultados de uma junção de grupo e você poderá acessar os itens, mas você não
poderá identificar a chave na qual eles correspondem. Portanto, geralmente há maior utilidade em
selecionar os resultados da junção de grupo em um novo tipo que também tenha o nome da chave,
conforme mostrado no exemplo anterior.
Além disso, é claro que você pode usar o resultado de uma junção de grupo como o gerador de outra
subconsulta:
var innerGroupJoinQuery2 =
from category in categories
join prod in products on category.ID equals prod.CategoryID into prodGroup
from prod2 in prodGroup
where prod2.UnitPrice > 2.50M
select prod2;
var leftOuterJoinQuery =
from category in categories
join prod in products on category.ID equals prod.CategoryID into prodGroup
from item in prodGroup.DefaultIfEmpty(new Product { Name = String.Empty, CategoryID = 0 })
select new { CatName = category.Name, ProdName = item.Name };
O operador equals
Uma cláusula join realiza uma junção por igualdade. Em outras palavras, você só pode basear
correspondências na igualdade de duas chaves. Não há suporte para outros tipos de comparações, como
"maior que" ou "não é igual a". Para se certificar de que todas as junções são junções por igualdade, a
cláusula join usa a palavra-chave equals em vez do operador == . A palavra-chave equals só pode
ser usada em uma cláusula join e ela difere do operador == em um aspecto importante. Com equals ,
a chave esquerda consome a sequência de origem externa e a chave direita consome a origem interna. A
origem externa somente está no escopo no lado esquerdo de equals e a sequência de origem interna
somente está no escopo no lado direito.
Chaves compostas
Você pode testar a igualdade de vários valores, usando uma chave de composição. Para obter mais
informações, consulte Unir usando chaves compostas. As chaves compostas também podem ser usadas
em uma cláusula group .
Exemplo
O exemplo a seguir compara os resultados de uma junção interna, uma junção de grupo e uma junção
externa esquerda nas mesmas fontes de dados, usando as mesmas chaves correspondentes. Foi
adicionado algum código extra nesses exemplos a fim de deixar os resultados na tela do console mais
claros.
class JoinDemonstration
{
#region Data
class Product
{
public string Name { get; set; }
public int CategoryID { get; set; }
}
class Category
{
public string Name { get; set; }
public int ID { get; set; }
}
app.InnerJoin();
app.GroupJoin();
app.GroupInnerJoin();
app.GroupJoin3();
app.LeftOuterJoin();
app.LeftOuterJoin2();
void InnerJoin()
{
// Create the query that selects
// a property from each element.
var innerJoinQuery =
from category in categories
join prod in products on category.ID equals prod.CategoryID
select new { Category = category.ID, Product = prod.Name };
Console.WriteLine("InnerJoin:");
// Execute the query. Access results
// with a simple foreach statement.
foreach (var item in innerJoinQuery)
{
Console.WriteLine("{0,-10}{1}", item.Product, item.Category);
}
Console.WriteLine("InnerJoin: {0} items in 1 group.", innerJoinQuery.Count());
Console.WriteLine(System.Environment.NewLine);
void GroupJoin()
{
// This is a demonstration query to show the output
// of a "raw" group join. A more typical group join
// is shown in the GroupInnerJoin method.
var groupJoinQuery =
from category in categories
join prod in products on category.ID equals prod.CategoryID into prodGroup
select prodGroup;
Console.WriteLine("Simple GroupJoin:");
void GroupInnerJoin()
{
var groupJoinQuery2 =
from category in categories
orderby category.ID
join prod in products on category.ID equals prod.CategoryID into prodGroup
select new
{
Category = category.Name,
Products = from prod2 in prodGroup
orderby prod2.Name
select prod2
};
//Console.WriteLine("GroupInnerJoin:");
int totalItems = 0;
Console.WriteLine("GroupInnerJoin:");
foreach (var productGroup in groupJoinQuery2)
{
Console.WriteLine(productGroup.Category);
foreach (var prodItem in productGroup.Products)
{
totalItems++;
Console.WriteLine(" {0,-10} {1}", prodItem.Name, prodItem.CategoryID);
}
}
Console.WriteLine("GroupInnerJoin: {0} items in {1} named groups", totalItems,
groupJoinQuery2.Count());
Console.WriteLine(System.Environment.NewLine);
}
void GroupJoin3()
{
var groupJoinQuery3 =
from category in categories
join product in products on category.ID equals product.CategoryID into prodGroup
from prod in prodGroup
orderby prod.CategoryID
select new { Category = prod.CategoryID, ProductName = prod.Name };
//Console.WriteLine("GroupInnerJoin:");
int totalItems = 0;
Console.WriteLine("GroupJoin3:");
foreach (var item in groupJoinQuery3)
{
totalItems++;
Console.WriteLine(" {0}:{1}", item.ProductName, item.Category);
}
void LeftOuterJoin()
{
// Create the query.
// Create the query.
var leftOuterQuery =
from category in categories
join prod in products on category.ID equals prod.CategoryID into prodGroup
select prodGroup.DefaultIfEmpty(new Product() { Name = "Nothing!", CategoryID =
category.ID });
void LeftOuterJoin2()
{
// Create the query.
var leftOuterQuery2 =
from category in categories
join prod in products on category.ID equals prod.CategoryID into prodGroup
from item in prodGroup.DefaultIfEmpty()
select new { Name = item == null ? "Nothing!" : item.Name, CategoryID = category.ID };
InnerJoin:
Cola 1
Tea 1
Mustard 2
Pickles 2
Carrots 3
Bok Choy 3
Peaches 5
Melons 5
InnerJoin: 8 items in 1 group.
Unshaped GroupJoin:
Group:
Cola 1
Tea 1
Group:
Group:
Mustard 2
Pickles 2
Group:
Carrots 3
Bok Choy 3
Group:
Group:
Peaches 5
Melons 5
Unshaped GroupJoin: 8 items in 5 unnamed groups
GroupInnerJoin:
Beverages
Cola 1
Tea 1
Condiments
Mustard 2
Pickles 2
Vegetables
Bok Choy 3
Carrots 3
Grains
Fruit
Melons 5
Peaches 5
GroupInnerJoin: 8 items in 5 named groups
GroupJoin3:
Cola:1
Tea:1
Mustard:2
Pickles:2
Carrots:3
Bok Choy:3
Peaches:5
Melons:5
GroupJoin3: 8 items in 1 group
Comentários
Uma cláusula join que não é seguida por into é convertida em uma chamada de método Join. Uma
cláusula join que é seguida por into é convertida em uma chamada de método GroupJoin.
Consulte também
Palavras-chave de Consulta (LINQ )
LINQ (Consulta Integrada à Linguagem)
Operações join
Cláusula group
Executar junções externas esquerdas
Executar junções internas
Executar junções agrupadas
Ordenar os resultados de uma cláusula join
Unir usando chaves compostas
Sistemas de banco de dados compatíveis para Visual Studio
Cláusula let (Referência de C#)
04/11/2019 • 2 minutes to read • Edit Online
Em uma expressão de consulta, às vezes é útil armazenar o resultado de uma subexpressão para usá-lo em
cláusulas subsequentes. É possível fazer isso com a palavra-chave let , que cria uma nova variável de intervalo
e a inicializa com o resultado da expressão fornecida. Depois de inicializado com um valor, a variável de
intervalo não pode ser usada para armazenar outro valor. No entanto, se a variável de intervalo mantiver um
tipo passível de consulta, ela poderá ser consultada.
Exemplo
No exemplo a seguir, let é usado de duas maneiras:
1. Para criar um tipo enumerável que pode ser pesquisado por si só.
2. Para permitir que a consulta chame ToLower apenas uma vez na variável de intervalo word . Sem usar
let , seria necessário chamar ToLower em cada predicado na cláusula where .
class LetSample1
{
static void Main()
{
string[] strings =
{
"A penny saved is a penny earned.",
"The early bird catches the worm.",
"The pen is mightier than the sword."
};
Consulte também
Referência de C#
Palavras-chave de Consulta (LINQ )
LINQ (Consulta Integrada à Linguagem)
Introdução a LINQ em C#
Manipular exceções em expressões de consultas
ascending (Referência de C#)
04/11/2019 • 2 minutes to read • Edit Online
A palavra-chave contextual ascending é usada na cláusula orderby em expressões de consulta para especificar
que a ordem de classificação é do menor para o maior. Como ascending é a ordem de classificação padrão, não é
necessário especificá-la.
Exemplo
O exemplo a seguir mostra o uso de ascending em uma cláusula orderby.
IEnumerable<string> sortAscendingQuery =
from vegetable in vegetables
orderby vegetable ascending
select vegetable;
Consulte também
Referência de C#
LINQ em C#
descending
descending (Referência de C#)
04/11/2019 • 2 minutes to read • Edit Online
A palavra-chave contextual descending é usada na cláusula orderby em expressões de consulta para especificar
que a ordem de classificação é do maior para o menor.
Exemplo
O exemplo a seguir mostra o uso de descending em uma cláusula orderby.
IEnumerable<string> sortDescendingQuery =
from vegetable in vegetables
orderby vegetable descending
select vegetable;
Consulte também
Referência de C#
LINQ em C#
ascending
on (Referência de C#)
23/10/2019 • 2 minutes to read • Edit Online
A palavra-chave contextual on é usada na cláusula join de uma expressão de consulta a fim de especificar a
condição de união.
Exemplo
O exemplo a seguir mostra o uso de on em uma cláusula join .
var innerJoinQuery =
from category in categories
join prod in products on category.ID equals prod.CategoryID
select new { ProductName = prod.Name, Category = category.Name };
Consulte também
Referência de C#
LINQ (Consulta Integrada à Linguagem)
equals (Referência de C#)
23/10/2019 • 2 minutes to read • Edit Online
A palavra-chave contextual equals é usada uma cláusula join em uma expressão de consulta a fim de comparar
os elementos de duas sequências. Para obter mais informações, consulte Cláusula join.
Exemplo
O exemplo a seguir mostra o uso da palavra-chave equals em uma cláusula join .
var innerJoinQuery =
from category in categories
join prod in products on category.ID equals prod.CategoryID
select new { ProductName = prod.Name, Category = category.Name };
Consulte também
LINQ (Consulta Integrada à Linguagem)
by (Referência de C#)
04/11/2019 • 2 minutes to read • Edit Online
A palavra-chave contextual by é usada na cláusula group de uma expressão de consulta para especificar como os
itens retornados devem ser agrupados. Para obter mais informações, consulte Cláusula group.
Exemplo
O exemplo a seguir mostra o uso da palavra-chave contextual by em uma cláusula group para especificar que os
alunos devem ser agrupados de acordo com a primeira letra do sobrenome.
Consulte também
LINQ em C#
in (Referência de C#)
23/10/2019 • 2 minutes to read • Edit Online
Consulte também
Palavras-chave do C#
Referência de C#
Operadores C# (Referência de C#)
29/11/2019 • 6 minutes to read • Edit Online
O C# oferece vários operadores predefinidos, compatíveis com os tipos internos. Por exemplo, os
operadores aritméticos executam operações aritméticas com operandos numéricos, já os operadores
lógicos boolianos executam operações lógicas com operandos bool. Determinados operadores podem ser
sobrecarregados. Com a sobrecarga de operador, você pode especificar o comportamento do operador
para os operandos de um tipo definido pelo usuário.
Em uma expressão, a precedência e a associação dos operadores determinam a ordem na qual as
operações são executadas. Você pode usar parênteses para alterar a ordem de avaliação imposta pela
prioridade e pela associação dos operadores.
Precedência do operador
Em uma expressão com vários operadores, os operadores com maior precedência são avaliados antes dos
operadores com menor precedência. No exemplo a seguir, a multiplicação é executada primeiro porque
tem uma precedência mais alta do que a adição:
var a = 2 + 2 * 2;
Console.WriteLine(a); // output: 6
Use parênteses para alterar a ordem de avaliação imposta pela precedência do operador:
var a = (2 + 2) * 2;
Console.WriteLine(a); // output: 8
A tabela a seguir lista os operadores C#, começando com a precedência mais alta até a mais baixa. Os
operadores em cada linha têm a mesma precedência.
x.y, x?.y, x?[y], f(x), a[i], x++, x--, new, typeof, checked, Primary
unchecked, default, nameof, delegate, sizeof, stackalloc, x-
>y
x * y, x / y, x % y Multiplicativo
x + y, x – y Aditivo
x == y, x != y Igualdade
x || y OR condicional
Associação de operador
Quando os operadores têm a mesma precedência, a associação dos operadores determina a ordem na
qual as operações são executadas:
Os operadores associativos esquerdos são avaliados na ordem da esquerda para a direita. Exceto para
os operadores de atribuição e os operadores de União nulo, todos os operadores binários são
associativos à esquerda. Por exemplo, a + b - c é avaliado como (a + b) - c .
Os operadores associativos direitos são avaliados na ordem da direita para a esquerda. Os operadores
de atribuição, os operadores de União nula e o operador condicional ?: são associativos à direita. Por
exemplo, x = y = z é avaliado como x = (y = z) .
Use parênteses para alterar a ordem de avaliação imposta pela associação de operador:
int a = 13 / 5 / 2;
int b = 13 / (5 / 2);
Console.WriteLine($"a = {a}, b = {b}"); // output: a = 1, b = 6
Avaliação do operando
Sem considerar a relação com a precedência e a associação de operadores, os operandos em uma
expressão são avaliados da esquerda para a direita. Os exemplos a seguir demonstram a ordem em que
os operadores e os operandos são avaliados:
a + b a, b, +
a + b * c a, b, c, *, +
EXPRESSÃO ORDEM DE AVALIAÇÃO
a / b + c * d a, b, /, c, d, *, +
a / (b + c) * d a, b, c, +, /, d, *
Normalmente, todos os operandos do operador são avaliados. No entanto, alguns operadores avaliam os
operandos condicionalmente. Ou seja, o valor do operando mais à esquerda de tal operador define if (ou
quais) outros operandos devem ser avaliados. Esses operadores são os operadores lógicos condicionais
and ( && ) e or ( || ) , os operadores de União nula ?? e ??= , os operadores condicionais nulos ?. e
?[] e o operador condicional ?: . Para obter mais informações, consulte a descrição de cada operador.
Especificação da linguagem C#
Para obter mais informações, confira a seção Operadores da Especificação da linguagem C#.
Consulte também
Referência de C#
Expressões
Operadores aritméticos (referência do C#)
30/10/2019 • 16 minutes to read • Edit Online
Esses operadores têm suporte de todos os tipos numéricos integral e de ponto flutuante .
Operador de incremento ++
O operador de incremento unário ++ incrementa seu operando em 1. O operando precisa ser uma variável,
um acesso de propriedade ou um acesso de indexador.
Há duas formas de suporte para o operador de incremento: o operador de incremento pós-fixado, x++ ,eo
operador de incremento pré-fixado, ++x .
Operador de incremento pós-fixado
O resultado de x++ é o valor de x antes da operação, como mostra o exemplo a seguir:
int i = 3;
Console.WriteLine(i); // output: 3
Console.WriteLine(i++); // output: 3
Console.WriteLine(i); // output: 4
double a = 1.5;
Console.WriteLine(a); // output: 1.5
Console.WriteLine(++a); // output: 2.5
Console.WriteLine(a); // output: 2.5
Operador de decremento --
O operador de decremento unário -- decrementa o operando em 1. O operando precisa ser uma variável, um
acesso de propriedade ou um acesso de indexador.
Há duas formas de suporte para o operador de decremento: o operador de decremento pós-fixado, x-- ,eo
operador de decremento pré-fixado, --x .
Operador de decremento pós-fixado
O resultado de x-- é o valor de x antes da operação, como mostra o exemplo a seguir:
int i = 3;
Console.WriteLine(i); // output: 3
Console.WriteLine(i--); // output: 3
Console.WriteLine(i); // output: 2
Operador de decremento de prefixo
O resultado de --x é o valor de x após a operação, como mostra o exemplo a seguir:
double a = 1.5;
Console.WriteLine(a); // output: 1.5
Console.WriteLine(--a); // output: 0.5
Console.WriteLine(a); // output: 0.5
Console.WriteLine(+4); // output: 4
Console.WriteLine(-4); // output: -4
Console.WriteLine(-(-4)); // output: 4
uint a = 5;
var b = -a;
Console.WriteLine(b); // output: -5
Console.WriteLine(b.GetType()); // output: System.Int64
Operador de multiplicação *
O operador de multiplicação * calcula o produto dos operandos:
Operador de divisão /
O operador de divisão / divide o operando à esquerda pelo operando à direita.
Divisão de inteiros
Para os operandos de tipos inteiros, o resultado do operador / é de um tipo inteiro e igual ao quociente dos
dois operandos arredondados para zero:
Para obter o quociente dos dois operandos como um número de ponto flutuante, use o tipo float , double ou
decimal :
Console.WriteLine(13 / 5.0); // output: 2.6
int a = 13;
int b = 5;
Console.WriteLine((double)a / b); // output: 2.6
Se um dos operandos é decimal , outro operando não pode ser float nem double , porque nem float ou
double é implicitamente conversível para decimal . Você deve converter explicitamente o operando float ou
double para o tipo decimal . Para obter mais informações sobre conversões entre tipos numéricos, consulte
conversões numéricas internas.
Operador de resto %
O operador de resto % calcula o resto após dividir o operando à esquerda pelo à direita.
Resto inteiro
Para os operandos de tipos inteiros, o resultado de a % b é o valor produzido por a - (a / b) * b . O sinal do
resto diferente de zero é o mesmo que o do operando à esquerda, conforme mostra o seguinte exemplo:
NOTE
Esse método de computação do resto é análogo ao usado para operandos inteiros, mas diferente da especificação IEEE
754. Se você precisar da operação restante que está em conformidade com a especificação IEEE 754, use o método
Math.IEEERemainder.
Para saber mais sobre o comportamento do operador % com operandos não finitos, confira a seção Operador
de restante da especificação da linguagem C#.
Para os operandos decimal , o operador de resto % é equivalente ao operador de resto do tipo
System.Decimal.
O seguinte exemplo demonstra o comportamento do operador de resto com os operandos de ponto flutuante:
Console.WriteLine(-5.2f % 2.0f); // output: -1.2
Console.WriteLine(5.9 % 3.1); // output: 2.8
Console.WriteLine(5.9m % 3.1m); // output: 2.8
Operador de adição +
O operador de adição + calcula a soma dos operandos:
Você também pode usar o operador + para a combinação de delegado e concatenação de cadeia de
caracteres. Para obter mais informações, confira o artigo Operadores + e += .
Operador de subtração -
O operador de subtração - subtrai o operando à direita do operando à esquerda:
Você também pode usar o operador - para a remoção de delegado. Para obter mais informações, confira o
artigo Operadores - e -= .
Atribuição composta
Para um operador binário op , uma expressão de atribuição composta do formato
x op= y
equivale a
x = x op y
a -= 4;
Console.WriteLine(a); // output: 10
a *= 2;
Console.WriteLine(a); // output: 20
a /= 4;
Console.WriteLine(a); // output: 5
a %= 3;
Console.WriteLine(a); // output: 2
Devido a promoções numéricas, o resultado da operação op pode não ser implicitamente conversível no tipo
T de x . Nesse caso, se op for um operador predefinido e o resultado da operação for explicitamente
convertido no tipo T de x , uma expressão de atribuição composta da forma x op= y será equivalente a
x = (T)(x op y) , exceto que x será avaliada apenas uma vez. O exemplo a seguir demonstra esse
comportamento:
byte a = 200;
byte b = 100;
var c = a + b;
Console.WriteLine(c.GetType()); // output: System.Int32
Console.WriteLine(c); // output: 300
a += b;
Console.WriteLine(a); // output: 44
Você também usa os operadores += e -= para assinar e cancelar a assinatura de um evento, respectivamente.
Para obter mais informações, confira Como assinar e cancelar a assinatura de eventos.
Operadores aritméticos binários são associativos à esquerda. Ou seja, os operadores com o mesmo nível de
precedência são avaliados da esquerda para a direita.
Use parênteses, () , para alterar a ordem de avaliação imposta pela precedência e pela capacidade de
associação do operador.
Para obter a lista completa C# de operadores ordenados por nível de precedência, consulte a seção precedência
de operador do artigo C# operadores .
int a = int.MaxValue;
int b = 3;
Para os operandos do tipo decimal , o estouro aritmético sempre gera uma OverflowException e a divisão por
zero sempre gera uma DivideByZeroException.
Erros de arredondamento
Devido a limitações gerais da representação de ponto flutuante de números reais e aritméticas de ponto
flutuante, erros de arredondamento podem ocorrer em cálculos com tipos de ponto flutuante. Ou seja, o
resultado produzido de uma expressão pode diferir do resultado matemático esperado. O seguinte exemplo
demonstra vários casos desse tipo:
double a = 0.1;
double b = 3 * a;
Console.WriteLine(b == 0.3); // output: False
Console.WriteLine(b - 0.3); // output: 5.55111512312578E-17
decimal c = 1 / 3.0m;
decimal d = 3 * c;
Console.WriteLine(d == 1.0m); // output: False
Console.WriteLine(d); // output: 0.9999999999999999999999999999
Para obter mais informações, consulte comentários nas páginas de referência System. Double, System. singleou
System. decimal .
Especificação da linguagem C#
Para obter mais informações, confira as seguintes seções da especificação da linguagem C#:
Operadores de incremento e decremento pós-fixados
Operadores de incremento e decremento pré-fixados
Operador unário de adição
Operador unário de subtração
Operador de multiplicação
Operador de divisão
Operador de resto
Operador de adição
Operador de subtração
Atribuição composta
Os operadores verificados e não verificados
Promoções numéricas
Consulte também
Referência de C#
Operadores do C#
System.Math
System.MathF
Numéricos no .NET
Operadores lógicos boolianos (referência do C#)
29/11/2019 • 13 minutes to read • Edit Online
bool SecondOperand()
{
Console.WriteLine("Second operand is evaluated.");
return true;
}
Para os operandos dos tipos numéricos inteiros, o operador de ^ computa o bit lógico Exclusive ou de seus
operandos.
Operador OR lógico |
O operador | computa o OR lógico de seus operandos. O resultado de x | y será true se x ou y for
avaliado como true . Caso contrário, o resultado será false .
O operador | avalia os dois operandos mesmo que o operando esquerdo seja avaliado como true , de modo
que o resultado da operação seja true , independentemente do valor do operando à direita.
No exemplo a seguir, o operando à direita do operador | é uma chamada de método, que é executada
independentemente do valor do operando à esquerda:
bool SecondOperand()
{
Console.WriteLine("Second operand is evaluated.");
return true;
}
O operador OR lógico condicional || também computa o OR lógico e seus operandos, mas não avalia o
operando à direita se o operando à esquerda for avaliado como true .
Para os operandos dos tipos numéricos inteiros, o operador de | computa o bit-a lógico ou de seus operandos.
bool SecondOperand()
{
Console.WriteLine("Second operand is evaluated.");
return true;
}
O operador AND lógico & também computa o AND lógico de seus operandos, mas sempre avalia os dois
operandos.
No exemplo a seguir, o operando à direita do operador || é uma chamada de método, que não é executada se
o operando à esquerda for avaliado como true :
bool SecondOperand()
{
Console.WriteLine("Second operand is evaluated.");
return true;
}
O operador OR lógico | também computa o OR lógico de seus operandos, mas sempre avalia os dois
operandos.
O comportamento desses operadores difere do comportamento típico do operador com tipos de valores
anuláveis. Normalmente, um operador que é definido para operandos de um tipo de valor também pode ser
usado com operandos do tipo de valor anulável correspondente. Esse operador produz null se qualquer um de
seus operandos for avaliado como null . No entanto, os operadores & e | podem produzir não nulo mesmo
que um dos operandos seja avaliado como null . Para obter mais informações sobre o comportamento do
operador com tipos de valores anuláveis, consulte a seção operadores levantados do artigo tipos de valores
anuláveis .
Você também pode usar os operadores ! e ^ com operandos bool? , como mostra o exemplo a seguir:
Atribuição composta
Para um operador binário op , uma expressão de atribuição composta do formato
x op= y
equivale a
x = x op y
exceto que x é avaliado apenas uma vez.
Os operadores & , | e ^ suportam a atribuição de compostos, conforme mostrado no exemplo a seguir:
test |= true;
Console.WriteLine(test); // output: True
test ^= false;
Console.WriteLine(test); // output: True
Precedência do operador
A lista a seguir ordena os operadores lógicos, começando da mais alta precedência até a mais baixa:
Operador de negação lógico !
Operador AND lógico &
Operador OR exclusivo lógico ^
Operador OR lógico |
Operador AND lógico condicional &&
Operador OR lógico condicional ||
Use parênteses, () , para alterar a ordem de avaliação imposta pela precedência do operador:
Para obter a lista completa C# de operadores ordenados por nível de precedência, consulte a seção precedência
de operador do artigo C# operadores .
Especificação da linguagem C#
Para obter mais informações, confira as seguintes seções da especificação da linguagem C#:
Operador de negação lógico
Operadores lógicos
Operadores lógicos condicionais
Atribuição composta
Consulte também
Referência de C#
Operadores do C#
Operadores shift e bit a bit
Operadores bit a bit e de deslocamento (referência
do C#)
27/11/2019 • 13 minutes to read • Edit Online
Os operadores a seguir executam operações de Shift ou or de bit com operandos dos tipos numéricos inteiros ou
do tipo Char :
Operador unário ~ (complemento bit a bit)
Operadores binários << (deslocamento à esquerda) e >> (deslocamento à direita)
Operadores binários & (AND lógico), | (OR lógico) e ^ (OR exclusivo lógico)
Esses operadores são definidos para os tipos int , uint , long e ulong . Quando ambos os operandos são de
outros tipos integrais ( sbyte , byte , short , ushort ou char ), seus valores são convertidos no tipo int , que
também é o tipo de resultado de uma operação. Quando os operandos são de tipos integrais diferentes, seus
valores são convertidos no tipo integral mais próximo que o contém. Para saber mais, confira a seção Promoções
numéricas da Especificação da linguagem C#.
Os operadores & , | e ^ também são definidos para operandos do tipo bool . Para obter mais informações,
veja Operadores lógicos boolianos.
As operações de deslocamento e bit a bit nunca causam estouro e produzem os mesmos resultados nos
contextos marcados e desmarcados.
uint a = 0b_0000_1111_0000_1111_0000_1111_0000_1100;
uint b = ~a;
Console.WriteLine(Convert.ToString(b, toBase: 2));
// Output:
// 11110000111100001111000011110011
Você também pode usar o símbolo ~ para declarar finalizadores. Para mais informações, consulte Finalizadores.
uint x = 0b_1100_1001_0000_0000_0000_0000_0001_0001;
Console.WriteLine($"Before: {Convert.ToString(x, toBase: 2)}");
uint y = x << 4;
Console.WriteLine($"After: {Convert.ToString(y, toBase: 2)}");
// Output:
// Before: 11001001000000000000000000010001
// After: 10010000000000000000000100010000
Como os operadores de deslocamento são definidos apenas para os tipos int , uint , long e ulong , o
resultado de uma operação sempre contém pelo menos 32 bits. Se o operando à esquerda for de outro tipo
integral ( sbyte , byte , short , ushort ou char ), seu valor será convertido no tipo int , como mostra o
exemplo a seguir:
byte a = 0b_1111_0001;
var b = a << 8;
Console.WriteLine(b.GetType());
Console.WriteLine($"Shifted byte: {Convert.ToString(b, toBase: 2)}");
// Output:
// System.Int32
// Shifted byte: 1111000100000000
Para saber mais sobre como o operando à direita do operador << define a contagem de deslocamento, veja a
seção Contagem de deslocamento dos operadores de deslocamento.
uint x = 0b_1001;
Console.WriteLine($"Before: {Convert.ToString(x, toBase: 2), 4}");
uint y = x >> 2;
Console.WriteLine($"After: {Convert.ToString(y, toBase: 2), 4}");
// Output:
// Before: 1001
// After: 10
As posições vazias de bit de ordem superior são definidas com base no tipo do operando à esquerda da seguinte
maneira:
Se o operando esquerdo for do tipo int ou long , o operador right-shift executará um deslocamento
aritmético : o valor do bit mais significativo (o bit de sinal) do operando esquerdo será propagado para as
posições de bits vazias de ordem superior. Ou seja, as posições vazias de bit de ordem superior são
definidas como zero se o operando à esquerda for positivo e definidas como um se ele for negativo.
int a = int.MinValue;
Console.WriteLine($"Before: {Convert.ToString(a, toBase: 2)}");
int b = a >> 3;
Console.WriteLine($"After: {Convert.ToString(b, toBase: 2)}");
// Output:
// Before: 10000000000000000000000000000000
// After: 11110000000000000000000000000000
Se o operando esquerdo for do tipo uint ou ulong , o operador right-shift executará um deslocamento
lógico : as posições de bit vazio de ordem superior são sempre definidas como zero.
uint c = 0b_1000_0000_0000_0000_0000_0000_0000_0000;
Console.WriteLine($"Before: {Convert.ToString(c, toBase: 2), 32}");
uint d = c >> 3;
Console.WriteLine($"After: {Convert.ToString(d, toBase: 2), 32}");
// Output:
// Before: 10000000000000000000000000000000
// After: 10000000000000000000000000000
Para saber mais sobre como o operando à direita do operador >> define a contagem de deslocamento, veja a
seção Contagem de deslocamento dos operadores de deslocamento.
uint a = 0b_1111_1000;
uint b = 0b_1001_1101;
uint c = a & b;
Console.WriteLine(Convert.ToString(c, toBase: 2));
// Output:
// 10011000
Para os operandos de bool , o operador & computa a lógica e seus operandos. O operador & unário é o
operador address-of.
uint a = 0b_1111_1000;
uint b = 0b_0001_1100;
uint c = a ^ b;
Console.WriteLine(Convert.ToString(c, toBase: 2));
// Output:
// 11100100
Operador OR lógico |
O operador | computa o OR lógico bit a bit de seus operandos:
uint a = 0b_1010_0000;
uint b = 0b_1001_0001;
uint c = a | b;
Console.WriteLine(Convert.ToString(c, toBase: 2));
// Output:
// 10110001
Atribuição composta
Para um operador binário op , uma expressão de atribuição composta do formato
x op= y
equivale a
x = x op y
uint a = 0b_1111_1000;
a &= 0b_1001_1101;
Display(a); // output: 10011000
a |= 0b_0011_0001;
Display(a); // output: 10111001
a ^= 0b_1000_0000;
Display(a); // output: 111001
a <<= 2;
Display(a); // output: 11100100
a >>= 4;
Display(a); // output: 1110
Devido a promoções numéricas, o resultado da operação op pode não ser implicitamente conversível no tipo T
de x . Nesse caso, se op for um operador predefinido e o resultado da operação for explicitamente convertido
no tipo T de x , uma expressão de atribuição composta da forma x op= y será equivalente a x = (T)(x op y) ,
exceto que x será avaliada apenas uma vez. O exemplo a seguir demonstra esse comportamento:
byte x = 0b_1111_0001;
int b = x << 8;
Console.WriteLine($"{Convert.ToString(b, toBase: 2)}"); // output: 1111000100000000
x <<= 8;
Console.WriteLine(x); // output: 0
Precedência do operador
A lista a seguir ordena os operadores lógicos, começando da mais alta precedência até a mais baixa:
Operador de complemento bit a bit ~
Operadores de deslocamento << e >>
Operador AND lógico &
Operador OR exclusivo lógico ^
Operador OR lógico |
Use parênteses, () , para alterar a ordem de avaliação imposta pela precedência do operador:
uint a = 0b_1101;
uint b = 0b_1001;
uint c = 0b_1010;
uint d1 = a | b & c;
Display(d1); // output: 1101
uint d2 = (a | b) & c;
Display(d2); // output: 1000
Para obter a lista completa C# de operadores ordenados por nível de precedência, consulte a seção precedência
de operador do artigo C# operadores .
int a = 0b_0001;
Console.WriteLine($"{a} << {count1} is {a << count1}; {a} << {count2} is {a << count2}");
// Output:
// 1 << 1 is 2; 1 << 225 is 2
int b = 0b_0100;
Console.WriteLine($"{b} >> {count1} is {b >> count1}; {b} >> {count2} is {b >> count2}");
// Output:
// 4 >> 1 is 2; 4 >> 225 is 2
especificação da linguagem C#
Para obter mais informações, confira as seguintes seções da especificação da linguagem C#:
Operador de complemento bit a bit
Operadores shift
Operadores lógicos
Atribuição composta
Promoções numéricas
Consulte também
Referência de C#
Operadores do C#
Operadores lógicos boolianos
Operadores de igualdade (Referência de C#)
30/10/2019 • 6 minutes to read • Edit Online
Operador de igualdade ==
O operador de igualdade == retornará true se seus operandos forem iguais; caso contrário, false .
Igualdade de tipos de valor
Os operandos dos tipos de valor internos serão iguais se seus valores forem iguais:
int a = 1 + 2 + 3;
int b = 6;
Console.WriteLine(a == b); // output: True
char c1 = 'a';
char c2 = 'A';
Console.WriteLine(c1 == c2); // output: False
Console.WriteLine(c1 == char.ToLower(c2)); // output: True
NOTE
Para os operadores == , < , > , <= e >= , se nenhum dos operandos for um número (Double.NaN ou Single.NaN), o
resultado da operação será false . Isso significa que o valor NaN não é superior, inferior nem igual a nenhum outro
valor double (ou float ), incluindo NaN . Para obter mais informações e exemplos, consulte o artigo de referência
Double.NaN ou Single.NaN.
Dois operandos do mesmo tipo de enum serão iguais se os valores correspondentes do tipo integral subjacente
forem iguais.
Os tipos struct definidos pelo usuário não dão suporte ao operador == , por padrão. Para dar suporte ao
operador == , um struct definido pelo usuário precisa sobrecarregá-lo.
A partir do C# 7.3, os operadores == e != são compatíveis com as tuplas do C#. Para obter mais informações,
consulte a seção Igualdade e tuplas do artigo Tipos de tupla do C#.
Igualdade de tipos de referência
Por padrão, dois operandos do tipo de referência são iguais quando se referem ao mesmo objeto:
public class ReferenceTypesEquality
{
public class MyClass
{
private int id;
Como mostra o exemplo, os tipos de referência definidos pelo usuário dão suporte ao operador == , por padrão.
No entanto, um tipo de referência pode sobrecarregar o operador == . Se um tipo de referência sobrecarregar o
operador == , use o método Object.ReferenceEquals para verificar se as duas referências desse tipo dizem
respeito ao mesmo objeto.
Igualdade da cadeia de caracteres
Dois operandos da cadeia de caracteres serão iguais quando ambos forem null ou ambas as instâncias da
cadeia de caracteres tiverem o mesmo comprimento e caracteres idênticos em cada posição de caractere:
string s1 = "hello!";
string s2 = "HeLLo!";
Console.WriteLine(s1 == s2.ToLower()); // output: True
string s3 = "Hello!";
Console.WriteLine(s1 == s3); // output: False
Essa é uma comparação ordinal que diferencia maiúsculas de minúsculas. Para obter mais informações sobre a
comparação de cadeias de caracteres, confira Como comparar cadeias de caracteres no C#.
Igualdade de delegado
Os dois operandos delegar do mesmo tipo de tempo de execução são iguais quando ambos são null ou suas
listas de invocação são do mesmo comprimento e tem entradas iguais em cada posição:
Action b = a + a;
Action c = a + a;
Console.WriteLine(object.ReferenceEquals(b, c)); // output: False
Console.WriteLine(b == c); // output: True
Operador de desigualdade !=
O operador de desigualdade != retornará true se seus operandos não forem iguais; caso contrário, false .
No caso dos operandos de tipos internos, a expressão x != y gera o mesmo resultado que a expressão
!(x == y) . Para obter mais informações sobre a igualdade de tipos, confira a seção Operador de igualdade.
int a = 1 + 1 + 2 + 3;
int b = 6;
Console.WriteLine(a != b); // output: True
string s1 = "Hello";
string s2 = "Hello";
Console.WriteLine(s1 != s2); // output: False
object o1 = 1;
object o2 = 1;
Console.WriteLine(o1 != o2); // output: True
Especificação da linguagem C#
Para obter mais informações, consulte a seção Operadores de teste de tipo e relacional na Especificação da
linguagem C#.
Consulte também
Referência de C#
Operadores do C#
System.IEquatable<T>
Object.Equals
Object.ReferenceEquals
Comparações de igualdade
Operadores de comparação
Operadores de comparação (referência do C#)
30/10/2019 • 3 minutes to read • Edit Online
Os operadores de comparação < (menor que), > (maior que), <= (menor ou igual) e >= ( maior que ou
igual), também conhecida como relacionais, comparam seus operandos. Esses operadores têm suporte de todos
os tipos numéricos integral e de ponto flutuante .
NOTE
Para os operadores == , < , > , <= e >= , se nenhum dos operandos for um número (Double.NaN ou Single.NaN), o
resultado da operação será false . Isso significa que o valor NaN não é superior, inferior nem igual a nenhum outro
valor double (ou float ), incluindo NaN . Para obter mais informações e exemplos, consulte o artigo de referência
Double.NaN ou Single.NaN.
Tipos de enumeração também dão suporte a operadores de comparação. No caso dos operandos do mesmo
tipo de enum, os valores correspondentes do tipo integral subjacente são comparados.
Os operadores == e != verificam se seus operandos são iguais ou não.
Especificação da linguagem C#
Para obter mais informações, consulte a seção Operadores de teste de tipo e relacional na Especificação da
linguagem C#.
Consulte também
Referência de C#
Operadores do C#
System.IComparable<T>
Operadores de igualdade
Operadores de acesso a membro (Referência de C#)
30/10/2019 • 11 minutes to read • Edit Online
using System.Collections.Generic;
Use . para formar um nome qualificado para acessar um tipo dentro de um namespace, como mostra o
código a seguir:
Use uma diretiva using para tornar o uso de nomes qualificados opcional.
Use . para acessar membros de tipo, estático e não-estático, como mostra o código a seguir:
Operador de indexador []
Os colchetes, [] , normalmente são usados para acesso de elemento de matriz, indexador ou ponteiro.
Acesso de matriz
O exemplo a seguir demonstra como acessar elementos de matriz:
int[] fib = new int[10];
fib[0] = fib[1] = 1;
for (int i = 2; i < fib.Length; i++)
{
fib[i] = fib[i - 1] + fib[i - 2];
}
Console.WriteLine(fib[fib.Length - 1]); // output: 55
Se um índice de matriz estiver fora dos limites da dimensão correspondente de uma matriz, uma
IndexOutOfRangeException será gerada.
Como mostra o exemplo anterior, você também usar colchetes quando declara um tipo de matriz ou instancia uma
instância de matriz.
Para obter mais informações sobre matrizes, confira Matrizes.
Acesso de indexador
O exemplo a seguir usa o tipo de Dictionary<TKey,TValue> .NET para demonstrar o acesso ao indexador:
Os indexadores permitem indexar instâncias de um tipo definido pelo usuário de maneira semelhante à indexação
de matriz. Ao contrário dos índices de matriz, que devem ser inteiros, os parâmetros do indexador podem ser
declarados como sendo de qualquer tipo.
Para obter mais informações sobre indexadores, confira Indexadores.
Outros usos de []
Para saber mais sobre o acesso a elemento de ponteiro, confira a seção Operador de acesso a elemento de
ponteiro [] do artigo Operadores relacionados a ponteiro.
Os colchetes também são usados para especificar atributos:
[System.Diagnostics.Conditional("DEBUG")]
void TraceMethod() {}
A?.B?.Do(C);
A?.B?[C];
O exemplo anterior também usa o operador de União nulo ?? para especificar uma expressão alternativa a ser
avaliada, caso o resultado de uma operação condicional nula seja null .
Invocação de delegado thread-safe
Use o operador ?. para verificar se um delegado é não nulo e chame-o de uma forma thread-safe (por exemplo,
quando você aciona um evento), conforme mostrado no código a seguir:
PropertyChanged?.Invoke(…)
Operador de invocação ()
Use parênteses, () , para chamar um método ou invocar um delegado.
O exemplo a seguir demonstra como chamar um método (com ou sem argumentos) e invocar um delegado:
Action<int> display = s => Console.WriteLine(s);
numbers.Clear();
display(numbers.Count); // output: 0
Você também pode usar parênteses ao invocar um construtor com o operador new .
Outros usos de ()
Você também pode usar parênteses para ajustar a ordem na qual as operações em uma expressão são avaliadas.
Para saber mais, confira Operadores C#.
Expressões de conversão, que executam conversões de tipo explícitas, também usam parênteses.
Operador de intervalo..
Disponível em C# 8,0 e posterior, o operador .. especifica o início e o fim de um intervalo de índices como
operandos. O operando à esquerda é um início inclusivo de um intervalo. O operando de lado direito é uma
extremidade exclusiva de um intervalo. Qualquer um dos operandos pode ser um índice do início ou do final de
uma sequência, como mostra o exemplo a seguir:
int[] numbers = new[] { 0, 10, 20, 30, 40, 50 };
int start = 1;
int amountToTake = 3;
int[] subset = numbers[start..(start + amountToTake)];
Display(subset); // output: 10 20 30
int margin = 1;
int[] inner = numbers[margin..^margin];
Display(inner); // output: 10 20 30 40
Como mostra o exemplo anterior, Expression a..b é do tipo System.Range. No Expression a..b , os resultados de
a e b devem ser conversíveis implicitamente para int ou Index.
Você pode omitir qualquer um dos operandos do operador .. para obter um intervalo aberto:
a.. equivale a a..^0
..b equivale a 0..b
.. equivale a 0..^0
Especificação da linguagem C#
Para obter mais informações, confira as seguintes seções da especificação da linguagem C#:
Acesso de membros
Acesso a elemento
Operador condicional nulo
Expressões de invocação
Para obter mais informações sobre índices e intervalos, consulte a Nota de proposta de recurso.
Consulte também
Referência de C#
Operadores do C#
?? (operador de união nula)
Operador ::
Operadores cast e teste de tipo (Referência de C#)
25/11/2019 • 11 minutes to read • Edit Online
Você pode usar os seguintes operadores para executar a verificação de tipo ou conversão de tipo:
operador is: para verificar se o tipo de runtime de uma expressão é compatível com um determinado tipo
operador as: para converter explicitamente uma expressão em um determinado tipo, se o tipo de runtime for
compatível com esse tipo
operador cast (): para executar uma conversão explícita
operador typeof: para obter a instância System.Type para um tipo
Operador is
O operador is verifica se o tipo de runtime de um resultado de expressão é compatível com um determinado
tipo. A partir C# do 7,0, o operador de is também testa um resultado de expressão em relação a um padrão.
A expressão com o operador is de teste de tipo tem o seguinte formato
E is T
em que E é uma expressão que retorna um valor e T é o nome de um tipo ou um parâmetro de tipo. E não
pode ser um método anônimo ou uma expressão lambda.
A expressão E is T retorna true se o resultado de E não for nulo e puder ser convertido para o tipo T por
uma conversão de referência, uma conversão boxing ou uma conversão unboxing; caso contrário, retorna false .
O operador is não considera conversões definidas pelo usuário.
O exemplo a seguir demonstra que o operador is retorna true se o tipo de runtime de um resultado da
expressão é derivado de um determinado tipo, ou seja, existe uma conversão de referência entre tipos:
O exemplo a seguir mostra que o operador de is leva em consideração as conversões boxing e unboxing, mas
não considera as conversões numéricas:
int i = 27;
Console.WriteLine(i is System.IFormattable); // output: True
object iBoxed = i;
Console.WriteLine(iBoxed is int); // output: True
Console.WriteLine(iBoxed is long); // output: False
Para saber mais sobre conversões em C#, confira o capítulo Conversões da Especificação da linguagem C#.
Teste de tipo com correspondência de padrões
A partir C# do 7,0, o operador de is também testa um resultado de expressão em relação a um padrão. Em
particular, ele dá suporte ao padrão de tipo da seguinte forma:
E is T v
em que E é uma expressão que retorna um valor, T é o nome de um tipo ou um parâmetro de tipo, e v é uma
nova variável local do tipo T . Se o resultado de E não for nulo e puder ser convertido para T por uma
conversão de referência, boxing ou unboxing, a expressão E is T v retornará true e o valor convertido do
resultado de E será atribuído à variável v .
O exemplo a seguir demonstra o uso do operador is com um padrão de tipo:
int i = 23;
object iBoxed = i;
int? jNullable = 7;
if (iBoxed is int a && jNullable is int b)
{
Console.WriteLine(a + b); // output 30
}
Para saber mais sobre o padrão de tipos e outros padrões com suporte, confira Correspondência de padrões com
is.
operador as
O operador as converte explicitamente o resultado de uma expressão para uma determinada referência ou tipo
de valor anulável. Se a conversão não for possível, o operador as retorna null . Ao contrário do operador cast (),
o operador as nunca lança uma exceção.
A expressão da forma
E as T
em que E é uma expressão que retorna um valor e T é o nome de um tipo ou um parâmetro de tipo, produz o
mesmo resultado que
E is T ? (T)(E) : (T)null
NOTE
Como mostrado no exemplo anterior, você precisa comparar o resultado da expressão as com null para verificar se uma
conversão foi bem-sucedida. A partir C# do 7,0, você pode usar o operador is para testar se a conversão é bem sucedido e,
se tiver sucesso, atribuir seu resultado a uma nova variável.
Operador cast ()
Uma expressão de conversão do formulário (T)E realiza uma conversão explícita do resultado da expressão E
para o tipo T . Se não existir nenhuma conversão explícita do tipo E para o tipo T , ocorrerá um erro em tempo
de compilação. No tempo de execução, uma conversão explícita pode não ter êxito e uma expressão de conversão
pode lançar uma exceção.
O exemplo a seguir demonstra conversões numéricas e de referência explícitas:
double x = 1234.7;
int a = (int)x;
Console.WriteLine(a); // output: 1234
Para saber mais sobre conversões explícitas sem suporte, confira a seção Conversões explícitas da Especificação
da linguagem C#. Para saber mais sobre como definir uma conversão de tipo explícito ou implícito personalizado,
confira Operadores de conversão definidos pelo usuário.
Outros usos de ()
Você também usa parênteses para chamar um método ou chamar um delegado .
Outro uso dos parênteses é ajustar a ordem na qual as operações em uma expressão são avaliadas. Para saber
mais, confira Operadores C#.
Operador typeof
O operador typeof obtém a instância System.Type para um tipo. O argumento do operador typeof deve ser o
nome de um tipo ou um parâmetro de tipo, como mostra o exemplo a seguir:
void PrintType<T>() => Console.WriteLine(typeof(T));
Console.WriteLine(typeof(List<string>));
PrintType<int>();
PrintType<System.Int32>();
PrintType<Dictionary<int, char>>();
// Output:
// System.Collections.Generic.List`1[System.String]
// System.Int32
// System.Int32
// System.Collections.Generic.Dictionary`2[System.Int32,System.Char]
Você também pode usar o operador typeof com tipos genéricos não associados. O nome de um tipo genérico
não associado deve conter o número apropriado de vírgulas, que é um a menos que o número de parâmetros de
tipo. O exemplo a seguir mostra o uso do operador typeof com um tipo genérico não associado:
Console.WriteLine(typeof(Dictionary<,>));
// Output:
// System.Collections.Generic.Dictionary`2[TKey,TValue]
Uma expressão não pode ser um argumento do operador typeof . Para obter a instância de System.Type para o
tipo de tempo de execução de um resultado de expressão, use o método Object.GetType.
Teste de tipo com o operador typeof
Use o operador typeof para verificar se o tipo de runtime do resultado da expressão corresponde exatamente a
um determinado tipo. O exemplo a seguir demonstra a diferença entre a verificação de tipo executada com o
operador typeof e o operador is:
Especificação da linguagem C#
Para obter mais informações, confira as seguintes seções da especificação da linguagem C#:
Operador is
Operador as
Expressões cast
Operador typeof
Consulte também
Referência de C#
Operadores do C#
Como converter com segurança usando correspondência de padrões e os operadores is e as
Genéricos no .NET
Operadores de conversão definidos pelo usuário
(Referência de C#)
30/10/2019 • 2 minutes to read • Edit Online
Um tipo definido pelo usuário pode definir uma conversão implícita ou explícita personalizada de outro tipo
ou para outro.
Conversões implícitas não requerem que uma sintaxe especial seja invocada e podem ocorrer em uma
variedade de situações, por exemplo, em atribuições e invocações de método. Conversões implícitas
predefinidas C# sempre são bem sucedidos e nunca geram uma exceção. Conversões implícitas definidas pelo
usuário devem se comportam dessa forma também. Se uma conversão personalizada puder gerar uma
exceção ou perder informações, defina-a como uma conversão explícita.
Conversões definidas pelo usuário não são consideradas pelos operadores is e as. Use o operador cast () para
invocar uma conversão explícita definida pelo usuário.
Use operator e as palavras-chave implicit ou explicit para definir uma conversão implícita ou explícita,
respectivamente. O tipo que define uma conversão deve ser um tipo de origem ou e destino dessa conversão.
Uma conversão entre os dois tipos definidos pelo usuário pode ser definida em qualquer um dos dois tipos.
O exemplo a seguir demonstra como definir uma conversão implícita e explícita:
using System;
byte number = d;
Console.WriteLine(number); // output: 7
Especificação da linguagem C#
Para obter mais informações, confira as seguintes seções da especificação da linguagem C#:
Operadores de conversão
Conversões Definidas pelo Usuário
Conversões implícitas
Conversões explícitas
Consulte também
Referência de C#
Operadores do C#
Sobrecarga de operador
Operadores cast e teste de tipo
Conversão e conversão de tipo
Conversões explícitas encadeadas definidas pelo usuário em C#
Operadores relacionados a ponteiro (referência do
C#)
30/10/2019 • 13 minutes to read • Edit Online
NOTE
Qualquer operação com ponteiros exige um contexto unsafe. O código que contém blocos não seguros deve ser compilado
com a opção do compilador -unsafe .
unsafe
{
int number = 27;
int* pointerToNumber = &number;
O operando do operador & deve ser uma variável fixa. Variáveis fixas são variáveis que residem em locais de
armazenamento não afetados pela operação do coletor de lixo. No exemplo anterior, a variável local number é
uma variável fixa, pois reside na pilha. Variáveis que residem em locais de armazenamento que podem ser
afetados pelo coletor de lixo (por exemplo, realocado) são chamadas de variáveis móveis. Campos de objeto e
elementos de matriz são exemplos de variáveis móveis. Você pode obter o endereço de uma variável móvel se
você "corrigir" ou "fixar", com uma instrução fixed . O endereço obtido é válido somente dentro do bloco de uma
instrução fixed . O exemplo a seguir mostra como usar uma instrução fixed e o operador & :
unsafe
{
byte[] bytes = { 1, 2, 3 };
fixed (byte* pointerToFirst = &bytes[0])
{
// The address stored in pointerToFirst
// is valid only inside this fixed statement block.
}
}
unsafe
{
char letter = 'A';
char* pointerToLetter = &letter;
Console.WriteLine($"Value of the `letter` variable: {letter}");
Console.WriteLine($"Address of the `letter` variable: {(long)pointerToLetter:X}");
*pointerToLetter = 'Z';
Console.WriteLine($"Value of the `letter` variable after update: {letter}");
}
// Output is similar to:
// Value of the `letter` variable: A
// Address of the `letter` variable: DCB977DDF4
// Value of the `letter` variable after update: Z
x->y
equivale a
(*x).y
unsafe
{
char* pointerToChars = stackalloc char[123];
NOTE
O operador de acesso de elemento de ponteiro não verifica se há erros fora dos limites.
Não é possível usar [] para acesso de elemento de ponteiro com uma expressão do tipo void* .
Você também pode usar o operador [] para acesso de indexador ou elemento de matriz.
unsafe
{
const int Count = 3;
int[] numbers = new int[Count] { 10, 20, 30 };
fixed (int* pointerToFirst = &numbers[0])
{
int* pointerToLast = pointerToFirst + (Count - 1);
Subtração de ponteiro
Para dois ponteiros p1 e p2 do tipo T* , a expressão p1 - p2 produz a diferença entre os endereços dados por
p1 e p2 divididos por sizeof(T) . O tipo de resultado é long . Ou seja, p1 - p2 é computado como
((long)(p1) - (long)(p2)) / sizeof(T) .
unsafe
{
int* numbers = stackalloc int[] { 0, 1, 2, 3, 4, 5 };
int* p1 = &numbers[1];
int* p2 = &numbers[5];
Console.WriteLine(p2 - p1); // output: 4
}
unsafe
{
int* numbers = stackalloc int[] { 0, 1, 2 };
int* p1 = &numbers[0];
int* p2 = p1;
Console.WriteLine($"Before operation: p1 - {(long)p1}, p2 - {(long)p2}");
Console.WriteLine($"Postfix increment of p1: {(long)(p1++)}");
Console.WriteLine($"Prefix increment of p2: {(long)(++p2)}");
Console.WriteLine($"After operation: p1 - {(long)p1}, p2 - {(long)p2}");
}
// Output is similar to
// Before operation: p1 - 816489946512, p2 - 816489946512
// Postfix increment of p1: 816489946512
// Prefix increment of p2: 816489946516
// After operation: p1 - 816489946516, p2 - 816489946516
Precedência do operador
A lista a seguir ordena operadores relacionados a ponteiro começando da precedência mais alta até a mais baixa:
Operadores de incremento x++ e decremento x-- sufixados e os operadores -> e []
Operadores de incremento ++x e decremento --x prefixados e os operadores & e *
Operadores de adição + e -
Operadores de comparação < , > , <= e >=
Operadores de igualdade == e !=
Use parênteses, () , para alterar a ordem de avaliação imposta pela precedência do operador.
Para obter a lista completa C# de operadores ordenados por nível de precedência, consulte a seção precedência
de operador do artigo C# operadores .
Especificação da linguagem C#
Para obter mais informações, confira as seguintes seções da especificação da linguagem C#:
Variáveis fixas e móveis
O operador address-of
Indireção de ponteiro
Acesso de membro do ponteiro
Acesso de elemento do ponteiro
Aritmética do ponteiro
Incrementar e decrementar ponteiros
Comparação de ponteiros
Consulte também
Referência de C#
Operadores do C#
Tipos de ponteiro
Palavra-chave unsafe
Palavra-chave fixed
Operador stackalloc
Operador sizeof
Operadores de atribuiçãoC# (referência)
30/10/2019 • 4 minutes to read • Edit Online
O operador de atribuição = atribui o valor do operando do lado direito a uma variável, uma propriedade ou um
elemento indexador fornecido pelo operando do lado esquerdo. O resultado de uma expressão de atribuição é o
valor atribuído a um operando do lado esquerdo. O tipo do operandos do lado direito deve ser do mesmo tipo
ou implicitamente conversível para o operando do lado esquerdo.
O operador de atribuição = é associativo à direita, ou seja, uma expressão do formulário
a = b = c
é avaliada como
a = (b = c)
O exemplo a seguir demonstra o uso do operador de atribuição com uma variável local, uma propriedade e um
elemento do indexador como seu operando esquerdo:
Console.WriteLine(numbers.Capacity);
numbers.Capacity = 100;
Console.WriteLine(numbers.Capacity);
// Output:
// 4
// 100
int newFirstElement;
double originalFirstElement = numbers[0];
newFirstElement = 5;
numbers[0] = newFirstElement;
Console.WriteLine(originalFirstElement);
Console.WriteLine(numbers[0]);
// Output:
// 1
// 5
No caso do operador de atribuição de referência, os dois operandos devem ser do mesmo tipo.
Atribuição composta
Para um operador binário op , uma expressão de atribuição composta do formato
x op= y
equivale a
x = x op y
Especificação da linguagem C#
Saiba mais na seção Operadores de atribuição na Especificação da linguagem C#.
Para obter mais informações sobre o operador de atribuição de referência = ref , consulte a Nota de proposta
de recurso.
Consulte também
Referência de C#
Operadores do C#
ref keyword
Operadores + e += (referência de C#)
25/11/2019 • 3 minutes to read • Edit Online
Os operadores de + e += têm suporte dos tipos numéricos de ponto flutuante e integral internos, o tipo de
cadeia de caracteres e os tipos delegados .
Para obter informações sobre o operador + aritmético, consulte as seções Operadores de adição e subtração
unários e Operador de adição + do artigo Operadores aritméticos.
A partir C# do 6, a interpolação de cadeia de caracteres fornece uma maneira mais conveniente de Formatar
cadeias:
Combinação de delegados
Para operandos do mesmo tipo delegado, o operador + retorna uma nova instância de delegado que, quando
invocada, chama o operando esquerdo e, em seguida, chama o operando direito. Se qualquer um dos operandos
for null , o operador + retornará o valor de outro operando (que também pode ser null ). O exemplo a seguir
mostra como os delegados podem ser combinados com o operador + :
x += y
equivale a
x = x + y
int i = 5;
i += 9;
Console.WriteLine(i);
// Output: 14
Console.WriteLine();
printer += () => Console.Write("b");
printer(); // output: ab
Você também usará o operador += para especificar um método de manipulador de eventos ao assinar um
evento. Para obter mais informações, confira Como assinar e cancelar a assinatura de eventos.
Especificação da linguagem C#
Para obter mais informações, veja as seções Operador de adição unário e Operador de adição da Especificação de
linguagem C#.
Consulte também
Referência de C#
Operadores do C#
Como concatenar várias cadeias de caracteres
Eventos
Operadores aritméticos
Operadores - e -=
Operadores - e -= (referência do C#)
30/10/2019 • 5 minutes to read • Edit Online
Os operadores de - e -= são suportados pelos tipos numéricos internos e de ponto flutuante e tipos
delegados .
Para obter informações sobre o operador - aritmético, consulte as seções Operadores de adição e subtração
unários e Operador de subtração - do artigo Operadores aritméticos.
Remoção de delegado
Para operandos do mesmo tipo delegado, o operador - retorna uma instância de delegado que é calculada da
seguinte maneira:
Se ambos os operandos forem não nulos e a lista de invocação do operando à direita for uma sublista
contígua apropriada da lista de invocação do operando à esquerda, o resultado da operação será uma
nova lista de invocação obtida removendo-se as entradas do operando à direita da lista de invocação do
operando à esquerda. Se a lista do operando à direita corresponder a várias sublistas contíguas na lista do
operando à esquerda, somente a sublista correspondente mais à direita será removida. Se a remoção
resultar em uma lista vazia, o resultado será null .
var abbaab = a + b + b + a + a + b;
abbaab(); // output: abbaab
Console.WriteLine();
var ab = a + b;
var abba = abbaab - ab;
abba(); // output: abba
Console.WriteLine();
Se a lista de invocação do operando à direita não for uma sublista contígua apropriada da lista de
invocação do operando à esquerda, o resultado da operação será o operando à esquerda. Por exemplo, a
remoção de um delegado que não faz parte do delegado multicast não tem consequências, e o delegado
multicast permanece inalterado.
Action a = () => Console.Write("a");
Action b = () => Console.Write("b");
var abbaab = a + b + b + a + a + b;
var aba = a + b + a;
O exemplo anterior também demonstra que, durante a remoção de delegados, as instâncias de delegado
são comparadas. Por exemplo, delegados produzidos pela avaliação de expressões lambda idênticas não
são iguais. Para saber mais sobre a igualdade de delegados, confira a seção Operadores de igualdade de
delegados da Especificação da linguagem C#.
Se o operando à esquerda for null , o resultado da operação será null . Se o operando à direita for
null , o resultado da operação será o operando à esquerda.
x -= y
equivale a
x = x - y
Console.WriteLine();
printer -= a;
printer(); // output: ab
Você também usará o operador -= para especificar um método de manipulador de eventos a remover ao
cancelar a assinatura de um evento. Para obter mais informações, confira Como assinar e cancelar a assinatura
de eventos.
Especificação da linguagem C#
Para obter mais informações, veja as seções Operador de subtração unário e Operador de subtração da
Especificação de linguagem C#.
Consulte também
Referência de C#
Operadores do C#
Eventos
Operadores aritméticos
Operadores + e +=
Operador ?: (referência do C#)
30/10/2019 • 4 minutes to read • Edit Online
O operador condicional ?: , também conhecido como operador condicional Ternário, avalia uma expressão
booliana e retorna o resultado de uma das duas expressões, dependendo se a expressão booliana é avaliada
como true ou false . Começando com C# 7.2, a expressão de referência condicional retorna a referência ao
resultado de uma de duas expressões.
A sintaxe do operador condicional é a seguinte:
A expressão condition deve ser avaliada para true ou false . Se condition for avaliada como true , a
expressão consequent será avaliada e seu resultado se tornará o resultado da operação. Se condition for
avaliada como false , a expressão alternative será avaliada e seu resultado se tornará o resultado da
operação. Somente consequent ou alternative é avaliada.
O tipo de consequent e alternative devem ser iguais ou deve haver uma conversão implícita de um tipo para
outro.
O operador condicional é associativo direito, ou seja, uma expressão da forma
a ? b : c ? d : e
é avaliada como
a ? b : (c ? d : e)
TIP
Você pode usar o seguinte dispositivo mnemônico para se lembrar de como o operador condicional é avaliado:
Console.WriteLine(sinc(0.1));
Console.WriteLine(sinc(0.0));
// Output:
// 0.998334166468282
// 1
Como o operador condicional original, a expressão condicional ref avalia apenas uma das duas expressões:
consequent ou alternative .
No caso da expressão condicional ref, o tipo de consequent e alternative devem ser iguais.
O exemplo a seguir demonstra o uso da expressão condicional ref:
int index = 7;
ref int refValue = ref ((index < 5) ? ref smallArray[index] : ref largeArray[index - 5]);
refValue = 0;
index = 2;
((index < 5) ? ref smallArray[index] : ref largeArray[index - 5]) = 100;
string classify;
if (input >= 0)
{
classify = "nonnegative";
}
else
{
classify = "negative";
}
Especificação da linguagem C#
Para saber mais, confira a seção Operador condicional na especificação da linguagem C#.
Para obter mais informações sobre a expressão ref condicional, consulte a Nota de proposta de recurso.
Consulte também
Referência de C#
Operadores do C#
Instrução if-else
Operadores ?. e ?[]
?? e?? = operadores
ref keyword
! (operador NULL-tolerante) (C# referência)
30/10/2019 • 4 minutes to read • Edit Online
Disponível em C# 8,0 e posterior, o operador sufixo unário ! é o operador NULL -tolerante. Em um contexto de
anotação anulávelhabilitado, você usa o operador NULL -tolerante para declarar que a expressão x de um tipo de
referência não é null : x! . O prefixo unário ! é o operador lógico de negação.
O operador NULL -tolerante não tem nenhum efeito no tempo de execução. Ele afeta apenas a análise de fluxo
estático do compilador, alterando o estado nulo da expressão. Em tempo de execução, a expressão x! é avaliada
como o resultado da expressão subjacente x .
Para obter mais informações sobre o recurso de tipos de referência anulável, consulte tipos de referência
anuláveis.
Exemplos
Um dos casos de uso do operador NULL -tolerante está no teste da lógica de validação de argumento. Por
exemplo, considere a seguinte classe:
#nullable enable
public class Person
{
public Person(string name) => Name = name ?? throw new ArgumentNullException(nameof(name));
Usando a estrutura de teste MSTest, você pode criar o seguinte teste para a lógica de validação no construtor:
[TestMethod, ExpectedException(typeof(ArgumentNullException))]
public void NullNameShouldThrowTest()
{
var person = new Person(null!);
}
Sem o operador NULL -tolerante, o compilador gera o seguinte aviso para o código anterior:
Warning CS8625: Cannot convert null literal to non-nullable reference type . Usando o operador NULL -tolerante,
você informa ao compilador que a passagem de null é esperada e não deve ser avisada sobre.
Você também pode usar o operador NULL -tolerante quando sabe definitivamente que uma expressão não pode
ser null , mas que o compilador não gerencia para reconhecê-la. No exemplo a seguir, se o método IsValid
retornar true , seu argumento não será null e você poderá desreferenciá-lo com segurança:
public static void Main()
{
Person? p = Find("John");
if (IsValid(p))
{
Console.WriteLine($"Found {p!.Name}");
}
}
Sem o operador NULL -tolerante, o compilador gera o seguinte aviso para o código p.Name :
Warning CS8602: Dereference of a possibly null reference .
Se você puder modificar o método IsValid , poderá usar o atributo NotNullWhen para informar ao compilador
que um argumento do método IsValid não pode ser null quando o método retornar true :
No exemplo anterior, você não precisa usar o operador NULL -tolerante porque o compilador tem informações
suficientes para descobrir que p não pode ser null dentro da instrução if . Para obter mais informações sobre
os atributos que permitem fornecer informações adicionais sobre o estado nulo de uma variável, consulte
Atualizar APIs com atributos para definir expectativas nulas.
Especificação da linguagem C#
Para obter mais informações, consulte a seção operador NULL -tolerante do rascunho da especificação de tipos de
referência anulável.
Consulte também
Referência de C#
Operadores do C#
Tutorial: design com tipos de referência anuláveis
?? e?? = OperatorsC# (referência)
08/11/2019 • 4 minutes to read • Edit Online
O operador de coalescência nula ?? retornará o valor do operando esquerdo se não for null ; caso
contrário, ele avaliará o operando direito e retornará seu resultado. O operador ?? não avaliará seu
operando direito se o operando esquerdo for avaliado como não nulo.
Disponível no C# 8,0 e posterior, o operador de atribuição de união nula ??= atribuirá o valor de seu
operando à direita para o operando à esquerda somente se o operando esquerdo for avaliado como null . O
operador ??= não avaliará seu operando direito se o operando esquerdo for avaliado como não nulo.
O operando do lado esquerdo do operador de ??= deve ser uma variável, uma Propriedadeou um elemento
de indexador .
Em C# 7,3 e anterior, o tipo do operando à esquerda do operador de ?? deve ser um tipo de referência ou
um tipo de valor anulável. A partir C# do 8,0, esse requisito é substituído pelo seguinte: o tipo do operando à
esquerda dos operadores de ?? e ??= não pode ser um tipo de valor não anulável. Em particular, começando
com C# o 8,0, você pode usar os operadores de União nula com parâmetros de tipo irrestrito:
a ?? b ?? c
d ??= e ??= f
a ?? (b ?? c)
d ??= (e ??= f)
Exemplos
Os operadores de ?? e ??= podem ser úteis nos seguintes cenários:
Em expressões com operadores condicionais nulos?. e? [], você pode usar o operador ?? para
fornecer uma expressão alternativa a ser avaliada caso o resultado da expressão com operações
condicionais nulas seja null :
Quando você trabalha com tipos de valor anulável e precisa fornecer um valor de um tipo de valor
subjacente, use o operador ?? para especificar o valor a ser fornecido em caso de um valor de tipo
anulável ser null :
int? a = null;
int b = a ?? -1;
Console.WriteLine(b); // output: -1
Use o método Nullable<T>.GetValueOrDefault() se o valor a ser usado quando um valor de tipo que
permite valor nulo for null tiver que ser o valor padrão do tipo de valor subjacente.
A partir C# do 7,0, você pode usar uma expressão throw como o operando à direita do operador ??
para tornar o código de verificação de argumento mais conciso:
O exemplo anterior também demonstra como usar membros aptos para expressão para definir uma
propriedade.
A partir C# do 8,0, você pode usar o operador de ??= para substituir o código do formulário
if (variable is null)
{
variable = expression;
}
Especificação da linguagem C#
Para obter mais informações sobre o operador de ?? , consulte a seção operador de União nulo da C#
especificação da linguagem.
Para obter mais informações sobre o operador de ??= , consulte a Nota de proposta de recurso.
Consulte também
Referência de C#
Operadores do C#
Operadores ?. e ?[]
Operador ?:
Operador => (referência do C#)
30/10/2019 • 3 minutes to read • Edit Online
O token de => tem suporte em duas formas: como o operador lambda e como um separador de um nome de
membro e a implementação de membro em uma definição de corpo de expressão.
Operador lambda
Em expressões lambda, o operador lambda => separa os parâmetros de entrada no lado esquerdo do corpo
lambda no lado direito.
O seguinte exemplo usa o recurso LINQ com a sintaxe de método para demonstrar o uso de expressões lambda:
int[] numbers = { 1, 4, 7, 10 };
int product = numbers.Aggregate(1, (interim, next) => interim * next);
Console.WriteLine(product); // output: 280
Os parâmetros de entrada de uma expressão lambda são fortemente tipados no momento da compilação.
Quando o compilador pode inferir os tipos de parâmetros de entrada, como no exemplo anterior, você pode
omitir declarações de tipo. Se você precisar especificar o tipo de parâmetros de entrada, deverá fazer isso para
cada parâmetro, como mostra o exemplo a seguir:
int[] numbers = { 1, 4, 7, 10 };
int product = numbers.Aggregate(1, (int interim, int next) => interim * next);
Console.WriteLine(product); // output: 280
O exemplo a seguir mostra como definir uma expressão lambda sem parâmetros de entrada:
em que expression é uma expressão válida. O tipo de retorno de expression deve ser implicitamente
conversível para o tipo de retorno do membro. Se o tipo de retorno do membro for void ou se o membro for
um construtor, um finalizador ou uma propriedade set acessador, expression deverá ser uma expressão de
instrução, que pode ser de qualquer tipo.
O seguinte exemplo mostra uma definição de corpo da expressão para um método Person.ToString :
As definições de corpo de expressão para métodos, operadores e propriedades somente leitura têm suporte a
C# partir de 6. As definições de corpo de expressão para construtores, finalizadores e acessadores de
propriedade e indexador têm C# suporte a partir de 7,0.
Para obter mais informações, consulte Membros aptos para expressão.
Especificação da linguagem C#
Para obter mais informações sobre o operador lambda, consulte a seção expressões de função anônimas da C#
especificação da linguagem.
Consulte também
Referência de C#
Operadores do C#
Operador :: (referência do C#)
30/10/2019 • 2 minutes to read • Edit Online
Use o qualificador de alias de namespace :: para acessar um membro de um namespace com alias. Você pode
usar o qualificador :: somente entre dois identificadores. O identificador à esquerda pode ser qualquer um
dos seguintes aliases:
Um alias de namespace criado com uma diretiva de alias using:
Um alias extern.
O alias global , que é o alias do namespace global. O namespace global é o namespace que contém
namespaces e tipos que não são declarados dentro de um namespace com nome. Quando usado com o
qualificador :: , o alias global sempre faz referência ao namespace global, mesmo se houver o alias de
namespace global definido pelo usuário.
O exemplo a seguir usa o alias global para acessar o namespace .NET System, que é um membro do
namespace global. Sem o alias global , o namespace System definido pelo usuário, que é um membro
do namespace MyCompany.MyProduct , seria acessado:
namespace MyCompany.MyProduct.System
{
class Program
{
static void Main() => global::System.Console.WriteLine("Using global alias");
}
class Console
{
string Suggestion => "Consider renaming this class";
}
}
NOTE
A palavra-chave global é o alias do namespace global apenas quando é o identificador à esquerda do
qualificador :: .
Você também pode usar o operador de . de acesso de membro para acessar um membro de um namespace
com alias. No entanto, o operador de . também é usado para acessar um membro de tipo. O qualificador ::
garante que o identificador à esquerda dele sempre faça referência a um alias de namespace, mesmo que exista
um tipo ou namespace com o mesmo nome.
Especificação da linguagem C#
Saiba mais na seção Qualificadores de alias de namespace da Especificação da linguagem C#.
Consulte também
Referência de C#
Operadores do C#
Usar namespaces
Operador await (referência de C#)
25/11/2019 • 6 minutes to read • Edit Online
using System;
using System.Net.Http;
using System.Threading.Tasks;
O exemplo anterior usa o método async Main , que é possível a C# partir de 7,1. Para obter mais
informações, confira a seção Operador await no método Main.
NOTE
Para obter uma introdução à programação assíncrona, confira Programação assíncrona com async e await. A
programação assíncrona com async e await segue o padrão assíncrono baseado em tarefas.
Use o operador await somente em um método, uma expressão lambda ou um método anônimo que
seja modificado pela palavra-chave async. Em um método assíncrono, não é possível usar o operador
await no corpo de uma função síncrona, dentro do bloco de uma instrução lock e em um contexto
desprotegido.
O operando await do operador geralmente é de um dos seguintes tipos .NET: Task, Task<TResult>,
ValueTask ou ValueTask<TResult>. No entanto, qualquer expressão aguardável pode ser o operando
do operador await . Para obter mais informações, confira a seção Expressões aguardáveis da
Especificação da linguagem C#.
A partir C# do 8,0, você pode usar a instrução await foreach para consumir um fluxo de dados
assíncrono. Para obter mais informações, consulte a seção fluxos assíncronos do artigo novidades no
C# 8,0 .
O tipo de expressão await t é TResult se o tipo de expressão t é Task<TResult> ou
ValueTask<TResult>. Se o tipo de t é Task ou ValueTask, o tipo de await t é void . Em ambos os
casos, se t gera uma exceção, await t gera a exceção novamente. Para obter mais informações
sobre o tratamento de exceções, confira a seção Exceções em métodos assíncronos do artigo Instrução
try-catch.
As palavras-chave async e await estão disponíveis em C# 5 e posteriores.
Especificação da linguagem C#
Para obter mais informações, confira a seção Expressões await da Especificação da linguagem C#.
Consulte também
Referência de C#
Operadores do C#
async
Modelo de programação assíncrona de tarefa
Programação assíncrona
Assincronia detalhada
Passo a passo: Como acessar a Web usando async e await
Tutorial: gerar e consumir fluxos assíncronos C# usando o 8,0 e o .net Core 3,0
operador padrão (referência do C#)
30/10/2019 • 2 minutes to read • Edit Online
O operador default produz o valor padrão de um tipo. O argumento para o operador default deve ser o
nome ou um parâmetro de tipo.
O exemplo a seguir mostra o uso do operador default :
Console.WriteLine(default(int)); // output: 0
Console.WriteLine(default(object) is null); // output: True
void DisplayDefaultOf<T>()
{
var val = default(T);
Console.WriteLine($"Default value of {typeof(T)} is {(val == null ? "null" : val.ToString())}.");
}
DisplayDefaultOf<int?>();
DisplayDefaultOf<System.Numerics.Complex>();
DisplayDefaultOf<System.Collections.Generic.List<int>>();
// Output:
// Default value of System.Nullable`1[System.Int32] is null.
// Default value of System.Numerics.Complex is (0, 0).
// Default value of System.Collections.Generic.List`1[System.Int32] is null.
Você também usa a palavra-chave default como o rótulo de caso padrão em uma instrução switch .
literal padrão
A partir do C# 7,1, você pode usar o literal default para produzir o valor padrão de um tipo quando o
compilador pode inferir o tipo de expressão. A expressão literal default produz o mesmo valor que a expressão
default(T) , em que T é o tipo inferido. Você pode usar o literal default em qualquer um dos seguintes casos:
Display(InitializeArray<int>(3)); // output: [ 0, 0, 0 ]
Display(InitializeArray<bool>(4, default)); // output: [ False, False, False, False ]
Especificação da linguagem C#
Para saber mais, confira a seção Expressões de valor padrão da Especificação da linguagem C#.
Para obter mais informações sobre o literal default , confira a nota da proposta do recurso.
Consulte também
Referência de C#
Operadores do C#
Tabela de valores padrão
Genéricos no .NET
operador delegate (referência do C#)
30/10/2019 • 2 minutes to read • Edit Online
O operador delegate cria um método anônimo que pode ser convertido em um tipo delegado:
NOTE
Começando com C# 3, as expressões lambda fornecem uma maneira mais concisa e expressiva de criar uma função
anônima. Use o operador => para construir uma expressão lambda:
Para saber mais sobre recursos de expressões lambda, por exemplo, que capturam variáveis externas, confira Expressões
lambda.
Você pode omitir a lista de parâmetros quando usa o operador delegate . Se você fizer isso, o método
anônimo criado poderá ser convertido em um tipo delegado com qualquer lista de parâmetros, como mostra o
exemplo a seguir:
// Output:
// Hello!
// This is world!
Essa é a única funcionalidade de métodos anônimos que não tem suporte por expressões lambda. Em todos os
outros casos, uma expressão lambda é a forma preferida de gravar código embutido.
Você também usa a palavra-chave delegate para declarar um tipo delegado.
Especificação da linguagem C#
Para obter mais informações, confira a seção Expressões de função anônima da Especificação da linguagem C#.
Consulte também
Referência de C#
Operadores do C#
= operador de >
operador nameof (referência do C#)
30/10/2019 • 2 minutes to read • Edit Online
O operador nameof obtém o nome de uma variável, tipo ou membro como uma cadeia de caracteres constante:
Como mostra o exemplo anterior, no caso de um tipo e um namespace, o nome produzido geralmente não é
totalmente qualificado.
O operador de nameof é avaliado em tempo de compilação e não tem nenhum efeito no tempo de execução.
Você pode usar o operador nameof para tornar o código de verificação de argumentos mais passível de
manutenção:
Especificação da linguagem C#
Para saber mais, confira a seção Expressões nameof da Especificação da linguagem C#.
Consulte também
Referência de C#
Operadores do C#
operador new (Referência em C#)
30/10/2019 • 3 minutes to read • Edit Online
Chamada de construtor
Para criar uma nova instância de um tipo, você normalmente invoca um dos construtores desse
tipo usando o operador new :
Você pode usar um inicializador de objeto ou coleção com o operador new para instanciar e
inicializar um objeto em uma instrução, como mostra o exemplo a seguir:
Criação de matriz
Você também usar o operador new para criar uma instância de matriz, como mostra o exemplo a
seguir:
Use a sintaxe de inicialização de matriz para criar uma instância de matriz e preenchê-la com os
elementos em uma instrução. O exemplo a seguir mostra várias maneiras de como fazer isso:
var a = new int[3] { 10, 20, 30 };
var b = new int[] { 10, 20, 30 };
var c = new[] { 10, 20, 30 };
Console.WriteLine(c.GetType()); // output: System.Int32[]
Especificação da linguagem C#
Para saber mais, confira a seção O operador new na especificação da linguagem C#.
Consulte também
Referência de C#
Operadores do C#
Inicializadores de objeto e de coleção
Operador sizeof (referência em C#)
30/10/2019 • 2 minutes to read • Edit Online
O operador sizeof retorna o número de bytes ocupados por uma variável de um determinado tipo. O
argumento do operador sizeof deve ser o nome de um tipo não gerenciado ou um parâmetro de tipo que seja
restrito a um tipo não gerenciado.
O operador sizeof exige um contexto não seguro. No entanto, as expressões apresentadas na tabela a seguir
são avaliadas em tempo de compilação para os valores constantes correspondentes e não exigem um contexto
não seguro:
sizeof(sbyte) 1
sizeof(byte) 1
sizeof(short) 2
sizeof(ushort) 2
sizeof(int) 4
sizeof(uint) 4
sizeof(long) 8
sizeof(ulong) 8
sizeof(char) 2
sizeof(float) 4
sizeof(double) 8
sizeof(decimal) 16
sizeof(bool) 1
Você também não precisará usar um contexto não seguro quando o operando do operador sizeof for o nome
de um tipo enumerado.
O exemplo a seguir demonstra o uso do operador sizeof :
using System;
unsafe
{
Console.WriteLine(sizeof(Point*)); // output: 8
}
}
O operador sizeof retorna o número de bytes que seriam alocados pelo Common Language Runtime na
memória gerenciada. Para tipos struct, esse valor inclui todo o preenchimento, como demonstra o exemplo
anterior. O resultado do operador sizeof pode ser diferente do resultado do método Marshal.SizeOf, que
retorna o tamanho de um tipo na memória não gerenciada.
Especificação da linguagem C#
Para obter mais informações, confira a seção O operador sizeof, nas especificações da linguagem C#.
Consulte também
Referência de C#
Operadores do C#
Operadores relacionados a ponteiro
Tipos de ponteiro
Tipos relativos a memória e extensão
Genéricos no .NET
Operador stackalloc (referência do C#)
30/10/2019 • 4 minutes to read • Edit Online
O operador stackalloc aloca um bloco de memória na pilha. Um bloco de memória alocado na pilha criado
durante a execução do método é descartado automaticamente quando esse método é retornado. Não é
possível liberar explicitamente a memória alocada com o operador stackalloc . Um bloco de memória
alocado de pilha não está sujeito à coleta de lixo e não precisa ser fixado com uma instrução fixed .
Na expressão stackalloc T[E] , T deve ser um tipo não gerenciado e E deve ser uma expressão do tipo
int .
Você pode atribuir o resultado do operador stackalloc a uma variável de um dos seguintes tipos:
A partir C# do 7,2, System.Span<T>ouSystem.ReadOnlySpan<T>, como mostra o exemplo a seguir:
int length = 3;
Span<int> numbers = stackalloc int[length];
for (var i = 0; i < length; i++)
{
numbers[i] = i;
}
Você não precisa usar um contexto unsafe quando atribui um bloco de memória alocado na pilha a
uma variável Span<T> ou ReadOnlySpan<T>.
Ao trabalhar com esses tipos, você pode usar uma expressão stackalloc em condicional ou
expressões de atribuição, como mostra o seguinte exemplo:
A partir C# do 8,0, você pode usar uma stackalloc expressão dentro de outras expressões sempre
que uma variávelSpan<T>ouReadOnlySpan<T>é permitida, como mostra o exemplo a seguir:
NOTE
É recomendável usar os tipos Span<T> ou ReadOnlySpan<T> sempre que possível para trabalhar com
memória alocada na pilha.
Como mostra o exemplo anterior, você precisa usar um contexto unsafe ao trabalhar com tipos de
ponteiro.
No caso de tipos de ponteiro, você pode usar uma stackalloc expressão somente em uma
declaração de variável local para inicializar a variável.
O conteúdo da memória recém-alocada é indefinido. A partir C# do 7,3, você pode usar a sintaxe do
inicializador de matriz para definir o conteúdo da memória alocada recentemente. O seguinte exemplo
demonstra várias maneiras de fazer isso:
Segurança
O uso de stackalloc habilita automaticamente os recursos de detecção de estouro de buffer no CLR
(Common Language Runtime). Se for detectada uma estouro de buffer, o processo será encerrado assim que
possível para minimizar a chance de o código mal-intencionado ser executado.
Especificação da linguagem C#
Para saber mais, confira a seção Alocação de pilha da Especificação da linguagem C#.
Consulte também
Referência de C#
Operadores do C#
Operadores relacionados a ponteiro
Tipos de ponteiro
Tipos relativos a memória e extensão
Operadores true e false (referência do C#)
29/11/2019 • 4 minutes to read • Edit Online
O operador true retorna o valor bool true para indicar que seu operando é definitivamente verdadeiro. O
operador false retorna o valor de bool true para indicar que seu operando é definitivamente falso. Não há
garantia de que os operadores true e false se complementarão. Ou seja, ambos os operadores true e false
podem retornar o valor bool false para o mesmo operando. Se um tipo define um dos dois operadores, ele
também deve definir outro operador.
TIP
Use o tipo de bool? , se você precisar dar suporte à lógica de três valores (por exemplo, ao trabalhar com bancos de
dados que dão suporte a um tipo booliano de três valores). C# fornece os operadores & e | que suportam a lógica de
três valores com os operandos bool? . Para obter mais informações, confira a seção Operadores lógicos booleanos
anuláveis do artigo Operadores lógicos boolianos.
expressões boolianas
Um tipo com o operador true definido pode ser o tipo de resultado de uma expressão condicional de controle
nas instruções if, do, while e for e no operador condicional ?: . Para saber mais, confira a seção Expressões
boolianas da Especificação da linguagem C#.
Exemplo
O exemplo a seguir apresenta o tipo que define os dois operadores, true e false . O tipo também sobrecarrega
os & lógicos e de operador de forma que o operador de && também possa ser avaliado para os operandos
desse tipo.
using System;
if (x == Yellow || y == Yellow)
{
return Yellow;
}
return Green;
}
public override bool Equals(object obj) => obj is LaunchStatus other && this == other;
public override int GetHashCode() => status;
}
Consulte também
Referência de C#
Operadores do C#
Sobrecarga de operador (referência de C#)
30/10/2019 • 6 minutes to read • Edit Online
Um tipo definido pelo usuário pode sobrecarregar um operador C# predefinido. Ou seja, um tipo pode
fornecer a implementação personalizada de uma operação caso um ou ambos os operandos sejam desse
tipo. A seção Operadores sobrecarregáveis mostra quais operadores do C# podem ser sobrecarregados.
Use a palavra-chave operator para declarar um operador. Uma declaração de operador deve satisfazer as
regras a seguir:
Ela inclui os modificadores public e static .
Um operador unário tem um parâmetro de entrada. Um operador binário tem dois parâmetros de
entrada. Em cada caso, pelo menos um parâmetro deve ter o tipo T ou T? , em que T é o tipo que
contém a declaração do operador.
O exemplo a seguir define uma estrutura simplificada para representar um número racional. A estrutura
sobrecarrega alguns dos operadores aritméticos:
using System;
Você pode estender o exemplo anterior definindo uma conversão implícita de int para Fraction . Em
seguida, os operadores sobrecarregados seriam compatíveis com os argumentos desses dois tipos. Ou
seja, tornaria-se possível adicionar um inteiro a uma fração e obter uma fração como um resultado.
Use também a palavra-chave operator para definir uma conversão de tipo personalizado. Para saber
mais, confira Operadores de conversão definidos pelo usuário.
Operadores sobrecarregáveis
A tabela a seguir fornece informações sobre capacidade de sobrecarga de operadores do C#:
+x, -x, !x, ~x, ++, --, true, false Esses operadores unários podem ser sobrecarregados.
+=, -=, *=, /=, %=, &=, |=, ^=, <<=, >>= Operadores de atribuição compostos não podem ser
sobrecarregados explicitamente. No entanto, quando um
operador binário estiver sobrecarregado, o operador de
atribuição composto correspondente, se houver, também
estará implicitamente sobrecarregado. Por exemplo, +=
é avaliado usando + , que pode ser sobrecarregado.
^ x, x = y, x. y, c? t: f, x?? y, x?? = y, x.. y, x-> y, =>, f (x), Esses operadores não podem ser sobrecarregados.
as, Await, marcada, desmarcada, padrão, delegado, é,
nameof, novo, sizeof, stackalloc, typeof
NOTE
Os operadores de comparação precisam ser sobrecarregados em pares. Ou seja, se o operador de um par está
sobrecarregado, o outro operador precisa estar sobrecarregado também. Esses pares são os seguintes:
Operadores == e !=
Operadores < e >
Operadores <= e >=
Especificação da linguagem C#
Para obter mais informações, confira as seguintes seções da especificação da linguagem C#:
Sobrecarga de operador
Operadores
Consulte também
Referência de C#
Operadores do C#
Operadores de conversão definidos pelo usuário
Por que os operadores sobrecarregados sempre são estáticos em C#?
Caracteres especiais de C#
31/10/2019 • 2 minutes to read • Edit Online
Os caracteres especiais são predefinidos e contextuais, e modificam o elemento de programa (uma cadeia de
caracteres literal, um identificador ou um nome de atributo) ao qual eles são acrescentados. O C# dá suporte aos
seguintes caracteres especiais:
@, o caractere identificador do texto.
$, o caractere da cadeia de caracteres interpolada.
Consulte também
Referência de C#
Guia de Programação em C#
interpolação de $-C# String (referência)
04/11/2019 • 8 minutes to read • Edit Online
O caractere especial $ identifica um literal de cadeia de caracteres como uma cadeia de caracteres
interpolada. Uma cadeia de caracteres interpolada é um literal de cadeia de caracteres que pode conter
expressões de interpolação. Quando uma cadeia de caracteres interpolada é resolvida em uma cadeia de
caracteres de resultado, itens com expressões de interpolação são substituídos pelas representações de
cadeia de caracteres dos resultados da expressão. Esse recurso está disponível a partir C# de 6.
A interpolação de cadeia de caracteres fornece uma sintaxe mais legível e conveniente para criar cadeias
de caracteres formatadas do que o recurso de formatação composta de cadeia de caracteres. O exemplo
a seguir usa os dois recursos para produzir o mesmo resultado:
// Composite formatting:
Console.WriteLine("Hello, {0}! Today is {1}, it's {2:HH:mm} now.", name, date.DayOfWeek, date);
// String interpolation:
Console.WriteLine($"Hello, {name}! Today is {date.DayOfWeek}, it's {date:HH:mm} now.");
// Both calls produce the same output that is similar to:
// Hello, Mark! Today is Wednesday, it's 19:40 now.
{<interpolationExpression>[,<alignment>][:<formatString>]}
Os elementos entre colchetes são opcionais. A seguinte tabela descreve cada elemento:
ELEMENTO DESCRIÇÃO
Console.WriteLine($"|{"Left",-7}|{"Right",7}|");
Caracteres especiais
Para incluir uma chave, "{" ou "}", no texto produzido por uma cadeia de caracteres interpolada, use duas
chaves, "{{" ou "}}". Para obter mais informações, consulte Chaves de escape.
Como os dois-pontos (":") têm um significado especial em um item de expressão de interpolação, para
usar um operador condicional em uma expressão de interpolação, coloque essa expressão entre
parênteses.
O seguinte exemplo mostra como incluir uma chave em uma cadeia de caracteres de resultado e como
usar um operador condicional em uma expressão de interpolação:
Uma cadeia de caracteres textual interpolada começa com o caractere de $ seguido pelo caractere de
@ . Para obter mais informações sobre cadeias de caracteres textuais, confira os tópicos cadeia de
caracteres e identificador textual.
NOTE
A partir C# do 8,0, você pode usar os tokens de $ e @ em qualquer ordem: $@"..." e @$"..." são cadeias de
caracteres textuais interpoladas válidas. Em versões C# anteriores, o token de $ deve aparecer antes do token
de @ .
System.Globalization.CultureInfo.CurrentCulture =
System.Globalization.CultureInfo.GetCultureInfo("nl-NL");
string messageInCurrentCulture = message.ToString();
Console.WriteLine($"{System.Globalization.CultureInfo.CurrentCulture,-10}
{messageInCurrentCulture}");
Console.WriteLine($"{specificCulture,-10} {messageInSpecificCulture}");
Console.WriteLine($"{"Invariant",-10} {messageInInvariantCulture}");
// Expected output is:
// nl-NL The speed of light is 299.792,458 km/s.
// en-IN The speed of light is 2,99,792.458 km/s.
// Invariant The speed of light is 299,792.458 km/s.
Recursos adicionais
Se não estiver familiarizado com a interpolação de cadeia de caracteres, confira o tutorial Interpolação de
cadeia de caracteres no C# Interativo. Você também pode verificar outro tutorial de interpolação de
cadeia de caracteres C# que demonstra como usar cadeias de caracteres interpoladas para produzir
cadeias de caracteres formatadas.
Compilação de cadeias de caracteres interpoladas
Se uma cadeia de caracteres interpolada tiver o tipo string , ela normalmente será transformada em
uma chamada de método String.Format. O compilador pode substituir String.Format por String.Concat
se o comportamento analisado for equivalente à concatenação.
Se uma cadeia de caracteres interpolada tiver o tipo IFormattable ou FormattableString, o compilador
gerará uma chamada para o método FormattableStringFactory.Create.
Especificação da linguagem C#
Para obter mais informações, consulte a seção cadeias de caracteres interpoladas da especificação da
linguagem C#.
Consulte também
Referência de C#
Caracteres especiais do C#
Cadeias de Caracteres
Tabela de formatação de resultados numéricos
Formatação de composição
String.Format
@ (Referência de C#)
31/10/2019 • 4 minutes to read • Edit Online
O caractere especial @ serve como um identificador textual. Ele pode ser usado das seguintes maneiras:
1. Para habilitar palavras-chave de C# a serem usadas como identificadores. O caractere @ atua como
prefixo de um elemento de código que o compilador deve interpretar como um identificador e não como
uma palavra-chave de C#. O exemplo a seguir usa o caractere @ para definir um identificador chamado
for que usa em um loop for .
2. Para indicar que um literal de cadeia de caracteres é interpretado de forma textual. O caractere @ neste
exemplo define um literal de cadeia de caracteres textual. Sequências de escape simples (como "\\"
para uma barra invertida), sequências de escape hexadecimais (como um "\x0041" para um A
maiúsculo) e sequências de escape Unicode (como "\u0041" para um A maiúsculo) são interpretadas de
forma textual. Somente uma sequência de escape de aspas ( "" ) não é interpretada literalmente; ela
produz aspas simples. Além disso, no caso de uma cadeia de caracteres interpolada textual, as sequências
de escape de chave ( {{ e }} ) não são interpretadas de forma textual; elas geram caracteres de chave
única. O exemplo a seguir define dois caminhos de arquivo idênticos, um usando um literal de cadeia de
caracteres regular e o outro usando um literal de cadeia de caracteres textual. Este é um dos usos mais
comuns de literais de cadeias de caracteres textuais.
Console.WriteLine(filename1);
Console.WriteLine(filename2);
// The example displays the following output:
// c:\documents\files\u0066.txt
// c:\documents\files\u0066.txt
O exemplo a seguir ilustra o efeito de definir um literal de cadeia de caracteres regular e um literal de
cadeia de caracteres textual que contêm sequências de caracteres idênticas.
Console.WriteLine(s1);
Console.WriteLine(s2);
// The example displays the following output:
// He said, "This is the last chance!"
// He said, "This is the last \u0063hance\x0021"
3. Para habilitar o compilador a distinguir entre os atributos em caso de um conflito de nomenclatura. Um
atributo é um tipo que deriva de Attribute. Seu nome de tipo normalmente inclui o sufixo Attribute,
embora o compilador não imponha essa convenção. O atributo pode, então, ser referenciado no código
por seu nome de tipo completo (por exemplo, [InfoAttribute] ou pelo nome abreviado (por exemplo,
[Info] ). No entanto, um conflito de nomenclatura ocorre se dois nomes de tipo abreviados forem
idênticos e um nome de tipo incluir o sufixo Attribute e o outro não incluir. Por exemplo, o código a
seguir não é compilado porque o compilador não consegue determinar se o atributo Info ou
InfoAttribute é aplicado à classe Example . Veja CS1614 para obter mais informações.
using System;
[AttributeUsage(AttributeTargets.Class)]
public class Info : Attribute
{
private string information;
[AttributeUsage(AttributeTargets.Method)]
public class InfoAttribute : Attribute
{
private string information;
[Info("A simple executable.")] // Generates compiler error CS1614. Ambiguous Info and InfoAttribute.
// Prepend '@' to select 'Info'. Specify the full name 'InfoAttribute' to select it.
public class Example
{
[InfoAttribute("The entry point.")]
public static void Main()
{
}
}
Consulte também
Referência de C#
Guia de Programação em C#
Caracteres especiais de C#
Diretivas de pré-processador do C#
23/10/2019 • 2 minutes to read • Edit Online
Consulte também
Referência de C#
Guia de Programação em C#
#if (Referência de C#)
29/11/2019 • 5 minutes to read • Edit Online
Quando o Compilador do Visual C# encontra uma diretiva #if , seguida eventualmente por uma diretiva
#endif, ele compila o código entre as diretivas somente quando o símbolo especificado é definido. Ao contrário
do C e do C++, não é possível atribuir um valor numérico a um símbolo. A instrução #if em C# é booliana e
testa apenas quando o símbolo foi definido ou não. Por exemplo:
#if DEBUG
Console.WriteLine("Debug version");
#endif
Você pode usar os operadores == (igualdade) e ! = (desigualdade) somente para testar os valores bool true
ou false . True significa que o símbolo foi definido. A instrução #if DEBUG tem o mesmo significado que
#if (DEBUG == true) . É possível usar os operadores && (e), || (ou) e ! (não) para avaliar se vários símbolos
foram definidos. Também é possível agrupar os símbolos e operadores com parênteses.
Comentários
#if , juntamente com as diretivas #else, #elif, #endif, #define e #undef, permite incluir ou excluir código com
base na existência de um ou mais símbolos. Isso pode ser útil ao compilar o código para um build de depuração
ou ao compilar para uma configuração específica.
Uma diretiva condicional que começa com uma diretiva #if deverá ser explicitamente encerrada com uma
diretiva #endif .
A diretiva #define permite definir um símbolo. Ao usar o símbolo como a expressão passada para a diretiva
#if , a expressão será avaliada como true .
Você também pode definir um símbolo com a opção do compilador -define. É possível excluir um símbolo com
#undef.
Um símbolo definido com -define ou com #define não entra em conflito com uma variável do mesmo nome.
Ou seja, um nome de variável não deve ser passado para uma diretiva de pré-processamento e um símbolo
pode ser avaliado apenas por uma diretiva de pré-processamento.
O escopo de um símbolo criado com #define é o arquivo no qual ele foi definido.
O sistema de compilação também reconhece os símbolos de pré-processador predefinidos que representam
estruturas de destino diferentes em projetos em estilo SDK. Eles são úteis ao criar aplicativos que podem
direcionar mais de uma implementação ou versão do .NET.
NOTE
Para projetos tradicionais de .NET Framework, você precisa configurar manualmente os símbolos de compilação
condicional para as diferentes estruturas de destino no Visual Studio por meio das páginas de propriedades do projeto.
As constantes TRACE e DEBUG são exemplos de outros símbolos predefinidos. Para substituir os valores
definidos no projeto, use a diretiva #define . Por exemplo, o símbolo DEBUG é definido automaticamente, de
acordo com as propriedades de configuração do build (Modo de Depuração ou Modo de Versão).
Exemplos
O exemplo a seguir mostra como definir um símbolo MYTEST em um arquivo e testar os valores dos símbolos
MYTEST e DEBUG. A saída deste exemplo depende do fato de você criar o projeto no modo de configuração de
versão ou no modo de configuração de depuração.
#define MYTEST
using System;
public class MyClass
{
static void Main()
{
#if (DEBUG && !MYTEST)
Console.WriteLine("DEBUG is defined");
#elif (!DEBUG && MYTEST)
Console.WriteLine("MYTEST is defined");
#elif (DEBUG && MYTEST)
Console.WriteLine("DEBUG and MYTEST are defined");
#else
Console.WriteLine("DEBUG and MYTEST are not defined");
#endif
}
}
O exemplo a seguir mostra como testar várias estruturas de destino para que você possa usar APIs mais
recentes, quando possível:
public class MyClass
{
static void Main()
{
#if NET40
WebClient _client = new WebClient();
#else
HttpClient _client = new HttpClient();
#endif
}
//...
}
Consulte também
Referência de C#
Guia de Programação em C#
Diretivas do pré-processador do C#
Como compilar condicionalmente com Trace e Debug
#else (Referência de C#)
23/10/2019 • 2 minutes to read • Edit Online
#else permite que você crie uma diretiva condicional composta, para que, caso nenhuma das expressões nas
diretivas anteriores #if ou #elif (opcional) seja avaliada como true , o compilador avalie todo o código entre
#else e o próximo #endif .
Comentários
#endif deve ser a próxima diretiva de pré-processador após #else . Consulte #if para obter um exemplo de como
usar #else .
Consulte também
Referência de C#
Guia de Programação em C#
Diretivas do pré-processador do C#
#elif (Referência de C#)
23/10/2019 • 2 minutes to read • Edit Online
O #elif permite criar uma diretiva condicional composta. A expressão #elif será avaliada se nenhum dos #if
anteriores nem nenhuma diretiva #elif , opcional, anterior avaliar expressões para true . Se uma expressão
#elif for avaliada como true , o compilador avalia todo o código entre o #elif e a próxima diretiva
condicional. Por exemplo:
#define VC7
//...
#if debug
Console.WriteLine("Debug build");
#elif VC7
Console.WriteLine("Visual Studio 7");
#endif
É possível usar os operadores == (igualdade), != (desigualdade), && (e) e || (ou), para avaliar vários
símbolos. Também é possível agrupar os símbolos e operadores com parênteses.
Comentários
#elif é equivalente a usar:
#else
#if
Usar #elif é mais simples, porque cada #if requer um #endif, enquanto um #elif pode ser usado sem um
#endif de correspondência.
Consulte #if para obter um exemplo de como usar #elif .
Consulte também
Referência de C#
Guia de Programação em C#
Diretivas do pré-processador do C#
#endif (Referência de C#)
23/10/2019 • 2 minutes to read • Edit Online
O #endif especifica o final de uma diretiva condicional, que começou com a diretiva #if. Por exemplo,
#define DEBUG
// ...
#if DEBUG
Console.WriteLine("Debug version");
#endif
Comentários
Uma diretiva condicional, começando com uma diretiva #if , deve ser encerrada explicitamente com uma
diretiva #endif . Consulte #if para obter um exemplo de como usar #endif .
Consulte também
Referência de C#
Guia de Programação em C#
Diretivas do pré-processador do C#
#define (Referência de C#)
23/10/2019 • 2 minutes to read • Edit Online
Use #define para definir um símbolo. Quando você usa o símbolo como a expressão passada para a diretiva #if,
a expressão será avaliada como true , conforme mostra o exemplo a seguir:
#define DEBUG
Comentários
NOTE
A diretiva #define não pode ser usada para declarar valores constantes como normalmente é feito em C e C++. As
constantes em C# são mais bem definidas como membros estáticos de uma classe ou struct. Se você tiver várias dessas
constantes, considere criar uma classe "Constantes" separada para guardá-las.
Os símbolos podem ser usados para especificar condições para compilação. É possível testar o símbolo com #if
ou #elif. Você também pode usar o ConditionalAttribute para executar uma compilação condicional.
É possível definir um símbolo, mas não é possível atribuir um valor a um símbolo. A diretiva #define deve ser
exibida no arquivo antes de usar as instruções que também não são diretivas de pré-processador.
Você também pode definir um símbolo com a opção do compilador -define. É possível excluir um símbolo com
#undef.
Um símbolo definido com -define ou com #define não entra em conflito com uma variável do mesmo nome.
Ou seja, um nome de variável não deve ser passado para uma diretiva de pré-processador e um símbolo apenas
pode ser avaliado por uma diretiva de pré-processador.
O escopo de um símbolo criado usando #define é o arquivo no qual o símbolo foi definido.
Como mostra o exemplo a seguir, é necessário colocar as diretivas #define na parte superior do arquivo.
#define DEBUG
//#define TRACE
#undef TRACE
using System;
#if (TRACE)
Console.WriteLine("Tracing is enabled.");
#endif
}
}
// Output:
// Debugging is enabled.
Consulte também
Referência de C#
Guia de Programação em C#
Diretivas do pré-processador do C#
const
Como: compilar condicionalmente com Trace e Debug
#undef
#if
#undef (Referência de C#)
23/10/2019 • 2 minutes to read • Edit Online
#undef permite excluir as definições de um símbolo, de modo que, usando o símbolo como a expressão em
uma diretiva #if, a expressão será avaliada como false .
Um símbolo pode ser definido com a diretiva #define ou com a opção do compilador /define. A diretiva #undef
deve ser exibida no arquivo antes de usar qualquer instrução que também não sejam diretivas.
Exemplo
// preprocessor_undef.cs
// compile with: /d:DEBUG
#undef DEBUG
using System;
class MyClass
{
static void Main()
{
#if DEBUG
Console.WriteLine("DEBUG is defined");
#else
Console.WriteLine("DEBUG is not defined");
#endif
}
}
Consulte também
Referência de C#
Guia de Programação em C#
Diretivas do pré-processador do C#
#warning (Referência de C#)
23/10/2019 • 2 minutes to read • Edit Online
#warning permite gerar um aviso do compilador CS1030 de nível um de um local específico no código. Por
exemplo:
Comentários
Um uso comum de #warning é em uma diretiva condicional. Também é possível gerar um erro definido pelo
usuário com #error.
Exemplo
// preprocessor_warning.cs
// CS1030 expected
#define DEBUG
class MainClass
{
static void Main()
{
#if DEBUG
#warning DEBUG is defined
#endif
}
}
Consulte também
Referência de C#
Guia de Programação em C#
Diretivas do pré-processador do C#
#error (Referência de C#)
23/10/2019 • 2 minutes to read • Edit Online
#error permite gerar um erro definido pelo usuário CS1029 de um local específico em seu código. Por exemplo:
Comentários
Um uso comum de #error é em uma diretiva condicional.
Também é possível gerar um aviso definido pelo usuário com #warning.
Exemplo
// preprocessor_error.cs
// CS1029 expected
#define DEBUG
class MainClass
{
static void Main()
{
#if DEBUG
#error DEBUG is defined
#endif
}
}
Consulte também
Referência de C#
Guia de Programação em C#
Diretivas do pré-processador do C#
#line (Referência de C#)
23/10/2019 • 4 minutes to read • Edit Online
O #line permite modificar o número de linha do compilador e (opcionalmente) a saída do nome de arquivo para
erros e avisos.
O exemplo a seguir mostra como relatar dois avisos associados aos números de linha. A diretiva #line 200 força
o próximo número de linha a ser 200 (embora o padrão seja #6) e, até a próxima diretiva #line , o nome de
arquivo será relatado como "Special". A diretiva #line default retorna a numeração de linhas à sua numeração
padrão, que conta as linhas que foram renumeradas pela diretiva anterior.
class MainClass
{
static void Main()
{
#line 200 "Special"
int i;
int j;
#line default
char c;
float f;
#line hidden // numbering not affected
string s;
double d;
}
}
Special(200,13): warning CS0168: The variable 'i' is declared but never used
Special(201,13): warning CS0168: The variable 'j' is declared but never used
MainClass.cs(9,14): warning CS0168: The variable 'c' is declared but never used
MainClass.cs(10,15): warning CS0168: The variable 'f' is declared but never used
MainClass.cs(12,16): warning CS0168: The variable 's' is declared but never used
MainClass.cs(13,16): warning CS0168: The variable 'd' is declared but never used
Comentários
A diretiva #line pode ser usada em uma etapa intermediária e automatizada no processo de build. Por exemplo,
se linhas fossem removidas do arquivo de código-fonte original, mas você ainda deseja que o compilador gere a
saída com base na numeração de linha original no arquivo, seria possível remover as linhas e, em seguida, simular
a numeração de linha original com #line .
A diretiva #line hidden oculta as linhas sucessivas do depurador, de modo que, quando o desenvolvedor
percorrer o código, quaisquer linhas entre um #line hidden e a próxima diretiva #line (supondo que não seja
outra diretiva #line hidden ) serão puladas. Essa opção também pode ser usada para permitir que o ASP.NET
diferencie entre o código gerado pelo computador e definido pelo usuário. Embora o ASP.NET seja o principal
consumidor desse recurso, é provável que mais geradores de origem façam uso dele.
A diretiva #line hidden não afeta os nomes de arquivo ou números de linha nos relatórios de erro. Ou seja, se um
erro for encontrado em um bloco oculto, o compilador reportará o nome do arquivo atual e o número de linha do
erro.
A diretiva #line filename especifica o nome de arquivo que você deseja que seja exibido na saída do compilador.
Por padrão, é usado o nome real do arquivo de código-fonte. O nome de arquivo deve estar entre aspas duplas ("")
e deve ser precedido por um número de linha.
Um arquivo de código-fonte pode ter qualquer número de diretivas #line .
Exemplo 1
O exemplo a seguir mostra como o depurador ignora as linhas ocultas no código. Quando você executar o
exemplo, ele exibirá três linhas de texto. No entanto, quando você definir um ponto de interrupção, conforme
mostrado no exemplo e apertar F10 para percorrer o código, você observará que o depurador ignorará a linha
oculta. Observe também que, mesmo que você defina um ponto de interrupção na linha oculta, o depurador ainda
a ignorará.
// preprocessor_linehidden.cs
using System;
class MainClass
{
static void Main()
{
Console.WriteLine("Normal line #1."); // Set break point here.
#line hidden
Console.WriteLine("Hidden line.");
#line default
Console.WriteLine("Normal line #2.");
}
}
Consulte também
Referência de C#
Guia de Programação em C#
Diretivas do pré-processador do C#
#region (Referência de C#)
23/10/2019 • 2 minutes to read • Edit Online
#region permite que você especifique um bloco de código que pode ser expandido ou recolhido ao usar o recurso
de estrutura de tópicos do editor do Visual Studio Code. Em arquivos de código mais longos, é conveniente
recolher ou ocultar uma ou mais regiões para que você possa se concentrar na parte do arquivo que está
trabalhando no momento. O exemplo a seguir mostra como definir uma região:
Comentários
Um bloco #region deverá ser encerrado com uma diretiva #endregion.
Um bloco #region não pode sobrepor um bloco #if. No entanto, um bloco #region pode ser aninhado em um
bloco #if e um bloco #if pode ser aninhado em um bloco #region .
Consulte também
Referência de C#
Guia de Programação em C#
Diretivas do pré-processador do C#
#endregion (Referência de C#)
23/10/2019 • 2 minutes to read • Edit Online
Consulte também
Referência de C#
Guia de Programação em C#
Diretivas do pré-processador do C#
#pragma (Referência de C#)
23/10/2019 • 2 minutes to read • Edit Online
O #pragma fornece ao compilador instruções especiais para a compilação do arquivo no qual ele é exibido. O
compilador deve dar suporte às instruções. Em outras palavras, não é possível usar #pragma para criar instruções
personalizadas de pré-processamento. O compilador Microsoft C# dá suporte a estas duas #pragma instruções:
#pragma warning
#pragma checksum
Sintaxe
#pragma pragma-name pragma-arguments
Parâmetros
pragma-name
O nome de um pragma reconhecido.
pragma-arguments
Argumentos específicos do pragma.
Consulte também
Referência de C#
Guia de Programação em C#
Diretivas do pré-processador do C#
#pragma warning
#pragma checksum
#pragma warning (Referência de C#)
23/10/2019 • 2 minutes to read • Edit Online
Sintaxe
#pragma warning disable warning-list
#pragma warning restore warning-list
Parâmetros
warning-list
Uma lista de números de aviso separada por vírgulas. O prefixo "CS" é opcional.
Quando não houver números de aviso especificados, o disable desabilita todos os avisos e o restore habilita
todos os avisos.
NOTE
Para localizar números de aviso no Visual Studio, compile o projeto e, em seguida, procure os números de aviso na janela de
Saída.
Exemplo
// pragma_warning.cs
using System;
Consulte também
Referência de C#
Guia de Programação em C#
Diretivas do pré-processador do C#
Erros do Compilador do C#
#pragma checksum (Referência de C#)
23/10/2019 • 2 minutes to read • Edit Online
Gera somas de verificação para os arquivos de origem para ajudar na depuração de páginas do ASP.NET.
Sintaxe
#pragma checksum "filename" "{guid}" "checksum bytes"
Parâmetros
"filename"
O nome do arquivo que exige o monitoramento de alterações ou atualizações.
"{guid}"
O GUID (identificador global exclusivo) do algoritmo de hash.
"checksum_bytes"
A cadeia de caracteres de dígitos hexadecimais que representa os bytes da soma de verificação. Deve ser um
número par de dígitos hexadecimais. Um número ímpar de dígitos resulta em um aviso em tempo de compilação
e a diretiva é ignorada.
Comentários
O depurador do Visual Studio usa uma soma de verificação para certificar-se de sempre encontrar a fonte correta.
O compilador calcula a soma de verificação para um arquivo de origem e, em seguida, emite a saída no arquivo
PDB (banco de dados do programa). Em seguida, o depurador usa o PDB para comparar com a soma de
verificação que ele calcula para o arquivo de origem.
Essa solução não funciona para projetos do ASP.NET, porque é a soma de verificação calculada é para o arquivo
de origem gerado e não para o arquivo .aspx. Para resolver esse problema, a #pragma checksum fornece suporte à
soma de verificação para páginas do ASP.NET.
Quando você cria um projeto do ASP.NET em Visual C#, o arquivo de origem gerado contém uma soma de
verificação para o arquivo .aspx, do qual a fonte é gerada. Então, o compilador grava essas informações no arquivo
PDB.
Se o compilador não encontrar uma diretiva #pragma checksum no arquivo, ele calcula a soma de verificação e
grava o valor no arquivo PDB.
Exemplo
class TestClass
{
static int Main()
{
#pragma checksum "file.cs" "{406EA660-64CF-4C82-B6F0-42D48172A799}" "ab007f1d23d9" // New checksum
}
}
Consulte também
Referência de C#
Guia de Programação em C#
Diretivas do pré-processador do C#
Opções do compilador de C#
25/11/2019 • 2 minutes to read • Edit Online
O compilador produz arquivos executáveis (.exe), bibliotecas de vínculo dinâmico (.dll) ou módulos de
código (.netmodule).
Todas as opções do compilador estão disponíveis em duas formas: -option e /option. A documentação
mostra apenas o formulário -option.
No Visual Studio, você define as opções do compilador no arquivo Web. config . Para obter mais
informações, consulte <compilador> Elemento.
Nesta seção
Criação de linha de comando com CSC. exe Informações sobre como criar um C# aplicativo Visual a
partir da linha de comando.
Como definir variáveis de ambiente para a linha de comando do Visual Studio Fornece as etapas
para executar o vsvars32. bat para habilitar as compilações de linha de comando.
Opções de compilador listadas por categoria C# Uma listagem categórica das opções do
compilador.
Opções de compilador listadas em ordem alfabética C# Uma listagem alfabética das opções do
compilador.
Seções relacionadas
Página de compilação, designer de projeto Definir propriedades que regem como o projeto é
compilado, compilado e depurado. Inclui informações sobre as etapas de build personalizadas em
projetos do Visual em C#.
Compilações padrão e personalizadas Informações sobre tipos de compilação e configurações.
Preparando e gerenciando compilações Procedimentos para a criação dentro do ambiente de
desenvolvimento do Visual Studio.
Build pela linha de comando com csc.exe
25/11/2019 • 7 minutes to read • Edit Online
Você pode invocar o compilador do C#, digitando o nome do seu arquivo executável (csc.exe) em um prompt de
comando.
Se você usar a janela do Prompt de Comando do Desenvolvedor do Visual Studio, todas as variáveis de
ambiente necessárias serão definidas para você. Para obter informações sobre como acessar essa ferramenta,
consulte o tópico Prompt de comando do desenvolvedor para o Visual Studio.
Se você usa uma janela de Prompt de Comando padrão, deve ajustar seu caminho antes de invocar o csc.exe de
qualquer subdiretório em seu computador. Você também deve executar o vsvars32.bat para definir as variáveis de
ambiente adequadas para dar suporte aos builds de linha de comando. Para obter mais informações sobre o
vsvars32. bat, incluindo instruções sobre como encontrá-lo e executá-lo, consulte como definir variáveis de
ambiente para a linha de comando do Visual Studio.
Se estiver trabalhando em um computador que tenha apenas o SDK (Software Development Kit) do Windows,
você poderá usar o compilador do C# no Prompt de Comando do SDK, que é aberto na opção de menu
Microsoft .NET Framework SDK.
Você também pode usar o MSBuild para compilar programas em C# programaticamente. Para mais informações,
consulte MSBuild.
O arquivo executável csc.exe normalmente está localizado na pasta Microsoft.NET\Framework\ <Versão> no
diretório Windows. O local pode variar dependendo da configuração exata de um computador específico. Se mais
de uma versão do .NET Framework estiver instalada em seu computador, você encontrará várias versões desse
arquivo. Para obter mais informações sobre essas instalações, consulte Determinando qual versão do .NET
Framework está instalada.
TIP
Quando você compila um projeto usando o IDE do Visual Studio, você pode exibir o comando csc e suas opções de
compilador associadas na janela Saída. Para exibir essas informações, siga as instruções em Como exibir, salvar e configurar
arquivos de log de build, para alterar o nível de detalhes dos dados de log para Normal ou Detalhado. Depois de
recompilar o projeto, pesquise na janela Saída por csc para localizar a invocação do compilador do C#.
Neste tópico
Regras para sintaxe de linha de comando
Linhas de comando de exemplo
Diferenças entre as saídas dos compiladores do C# e do C++
csc File.cs
Compila todos os arquivos do C# no diretório atual, com otimizações habilitadas e define o símbolo de
DEPURAÇÃO. A saída é File2.exe:
Compila todos os arquivos do C# no diretório atual, produzindo uma versão de depuração de File2.dll.
Logotipos e avisos não são exibidos:
Consulte também
Opções do compilador de C#
Opções do compilador de C# listadas em ordem alfabética
Opções do compilador de C# listadas por categoria
Main() e argumentos de linha de comando
Argumentos de linha de comando
Como exibir argumentos de linha de comando
Valores de retorno de Main()
Como definir variáveis de ambiente para a linha de
comando do Visual Studio
25/11/2019 • 2 minutes to read • Edit Online
O arquivo VsDevCmd.bat define as variáveis de ambiente adequadas para habilitar builds de linha de comando.
NOTE
O arquivo VsDevCmd.bat é um novo arquivo fornecido com o Visual Studio 2017. O Visual Studio 2015 e versões
anteriores usavam o VSVARS32.bat para a mesma finalidade. Esse arquivo era armazenado em \Arquivos de
Programas\Microsoft Visual Studio\Versão\Common7\Tools ou Arquivos de Programas (x86)\Microsoft Visual
Studio\Versão\Common7\Tools.
Se a versão atual do Visual Studio estiver instalada em um computador que também tiver uma versão anterior
do Visual Studio, você não deverá executar VsDevCmd.bat e VSVARS32.bat de versões diferentes na mesma
janela do Prompt de Comando. Em vez disso, você deve executar o comando para cada versão em sua própria
janela.
Para executar o VsDevCmd.BAT
1. No menu Iniciar, abra o Prompt de Comando do Desenvolvedor para VS 2017. Ele está na pasta do
Visual Studio 2017.
2. Altere para o subdiretório \Arquivos de Programas\Microsoft Visual
Studio\Versão\Oferta\Common7\Tools ou \Arquivos de Programas (x86)\Microsoft Visual
Studio\Versão\Oferta\Common7\Tools da sua instalação. (a Versão é 2017 para a versão atual. Oferta é
uma entre Enterprise, Professional ou Community).
3. Execute o VsDevCmd.bat digitando VsDevCmd.
Cau t i on
O VsDevCmd.bat pode variar de um computador para outro. Não substitua um arquivo VsDevCmd.bat
não encontrado ou danificado por um VsDevCmd.bat de outro computador. Em vez disso, execute
novamente a instalação para substituir o arquivo não encontrado.
Opções disponíveis para o VsDevCmd.BAT
Para ver as opções disponíveis para VsDevCmd.BAT, execute o comando com a opção -help :
VsDevCmd.bat -help
Consulte também
Build pela linha de comando com csc.exe
Opções do compilador de C# listadas por categoria
25/11/2019 • 7 minutes to read • Edit Online
As opções do compilador a seguir são classificadas por categoria. Para obter uma lista alfabética, consulte Opções
do compilador C# listadas em ordem alfabética.
Otimização
OPÇÃO FINALIDADE
Arquivos de saída
OPÇÃO FINALIDADE
-publicsign Aplicar uma chave pública sem assinar o assembly, mas definir
o bit no assembly indicando que o assembly está assinado.
Verificação de depuração/erros
OPÇÃO FINALIDADE
Pré-processador
OPÇÃO FINALIDADE
Recursos
OPÇÃO FINALIDADE
Diversos
OPÇÃO FINALIDADE
Opções obsoletas
OPÇÃO FINALIDADE
Consulte também
Opções do compilador de C#
Opções do compilador de C# listadas em ordem alfabética
Como definir variáveis de ambiente para a linha de comando do Visual Studio
Opções do compilador de C# listadas em ordem
alfabética
25/11/2019 • 7 minutes to read • Edit Online
As opções do compilador a seguir estão em ordem alfabética. Para obter uma lista categórica, consulte Opções
de Compilador de C# Listadas por Categoria.
OPÇÃO FINALIDADE
-bugreport Cria um arquivo 'Bug Report'. Esse arquivo será enviado junto
com informações de falha se for usado com -
errorreport:prompt ou -errorreport:send.
Consulte também
Opções do compilador de C#
Opções do compilador de C# listadas por categoria
Como definir variáveis de ambiente para a linha de comando do Visual Studio
<compiler> Element
@ (Opções do compilador de C#)
23/10/2019 • 2 minutes to read • Edit Online
A opção @ possibilita especificar um arquivo que contém opções do compilador e arquivos de código-fonte a
serem compilados.
Sintaxe
@response_file
Arguments
response_file
Um arquivo que lista as opções do compilador ou os arquivos de código-fonte a serem compilados.
Comentários
As opções do compilador e os arquivos de código-fonte serão processados pelo compilador apenas se eles
tiverem sido especificados na linha de comando.
Para especificar mais de um arquivo de resposta em uma compilação, especifique várias opções de arquivo de
resposta. Por exemplo:
@file1.rsp @file2.rsp
Em um arquivo de resposta, várias opções de compilador e de arquivos de código-fonte podem ser exibidas em
uma linha. Uma única especificação de opção do compilador deve ser exibida em uma linha (não é possível
abranger várias linhas). Os arquivos de resposta podem ter comentários que começam com o símbolo #.
Especificar opções do compilador de dentro de um arquivo de resposta é como emitir esses comandos na linha de
comando. Consulte Compilando da linha de comando para obter mais informações.
O compilador processa as opções de comando como elas são encontradas. Portanto, os argumentos da linha de
comando podem substituir opções listadas anteriormente em arquivos de resposta. Por outro lado, opções em um
arquivo de resposta substituirão as opções listadas anteriormente na linha de comando ou em outros arquivos de
resposta.
O C# fornece o arquivo csc.rsp, localizado no mesmo diretório que o arquivo csc.exe. Consulte -noconfig para
obter informações sobre csc.rsp.
Essa opção do compilador não pode ser definida no ambiente de desenvolvimento do Visual Studio nem pode ser
alterada por meio de programação.
Exemplo
A seguir, há algumas linhas de um exemplo de arquivo de resposta:
Essa opção adiciona um módulo criado com a opção target:module para a compilação atual.
Sintaxe
-addmodule:file[;file2]
Arguments
file , file2
Um arquivo de saída que contém metadados. O arquivo não pode conter um manifesto do assembly. Para
importar mais de um arquivo, separe os nomes de arquivo com vírgula ou ponto e vírgula.
Comentários
Todos os módulos adicionados com -addmodule devem estar no mesmo diretório que o arquivo de saída em
tempo de execução. Ou seja, é possível especificar um módulo em qualquer diretório em tempo de compilação,
mas o módulo deve estar no diretório do aplicativo em tempo de execução. Se o módulo não estiver no diretório
do aplicativo em tempo de execução, um TypeLoadException será obtido.
file não pode conter um assembly. Por exemplo, se o arquivo de saída foi criado com -target:module, seus
metadados podem ser importados com -addmodule.
Se o arquivo de saída tiver sido criado com uma opção -target diferente de -target:module, seus metadados
não poderão ser importados com o -addmodule, mas poderão ser importados com -reference.
Essa opção do compilador não está disponível no Visual Studio; um projeto não pode referenciar um módulo.
Além disso, essa opção do compilador não pode ser alterada programaticamente.
Exemplo
Compile o arquivo de origem input.cs e adicione metadados de metad1.netmodule e metad2.netmodule para
produzir out.exe :
Consulte também
Opções do compilador de C#
Gerenciando propriedades de solução e de projeto
Assemblies de vários arquivos
Como: Criar um assembly de vários arquivos
-appconfig (opções do compilador C#)
23/10/2019 • 3 minutes to read • Edit Online
Sintaxe
-appconfig:file
Arguments
file
Necessário. O arquivo de configuração de aplicativo que contém as configurações de associação de assembly.
Comentários
O -appconfig pode ser usado em cenários avançados, nos quais é necessário que um assembly referencie, ao
mesmo temo, tanto a versão do .NET Framework quanto a versão do .NET Framework para a versão do Silverlight
de um assembly de referência específico. Por exemplo, um designer XAML gravado no Windows Presentation
Foundation (WPF ) pode ter que referenciar a Área de Trabalho do WPF, para a interface do usuário do designer e
o subconjunto do WPF incluído no Silverlight. O mesmo assembly do designer deve acessar ambos os assemblies.
Por padrão, as referências separadas causam um erro do compilador, pois a associação de assembly considera os
dois assemblies equivalentes.
A opção do compilador -appconfig permite a especificação do local de um arquivo app.config que desabilita o
comportamento padrão por meio de uma marcação <supportPortability> , conforme mostrado no exemplo a
seguir.
<supportPortability PKT="7cec85d7bea7798e" enable="false"/>
NOTE
Caso o MSBuild (Microsoft Build Engine) seja usado para compilar o aplicativo, é possível definir a opção do compilador -
appconfig adicionando uma marcação de propriedade ao arquivo .csproj. Para usar o arquivo app.config que já está definido
no projeto, adicione a marca de propriedade <UseAppConfigForCompiler> ao arquivo .csproj e defina seu valor para true .
Para especificar um arquivo app.config diferente, adicione a marca de propriedade <AppConfigForCompiler> e defina seu
valor para o local do arquivo.
Exemplo
O exemplo a seguir mostra um arquivo app.config que habilita um aplicativo a referenciar as implementações do
.NET Framework e do .NET Framework para Silverlight de qualquer assembly do .NET Framework que exista em
ambas as implementações. A opção do compilador -appconfig especifica o local desse arquivo app.config.
<configuration>
<runtime>
<assemblyBinding>
<supportPortability PKT="7cec85d7bea7798e" enable="false"/>
<supportPortability PKT="31bf3856ad364e35" enable="false"/>
</assemblyBinding>
</runtime>
</configuration>
Consulte também
Elemento <supportPortability>
Opções do compilador de C# listadas em ordem alfabética
-baseaddress (opções do compilador C#)
23/10/2019 • 2 minutes to read • Edit Online
A opção -baseaddress permite especificar o endereço básico preferido em que uma DLL será carregada. Para
obter mais informações sobre quando e por que usar essa opção, consulte o Blog do Larry Osterman.
Sintaxe
-baseaddress:address
Arguments
address
O endereço básico da DLL. Esse endereço pode ser especificado como um número decimal, hexadecimal ou octal.
Comentários
O endereço básico padrão de uma DLL é definido pelo Common Language Runtime do .NET Framework.
Lembre-se de que a palavra de ordem inferior nesse endereço será arredondada. Por exemplo, se 0x11110001 for
especificado, será arredondado para 0x11110000.
Para concluir o processo de assinatura de uma DLL, use SN.EXE com a opção -R.
Para definir esta opção do compilador no ambiente de desenvolvimento do Visual Studio
1. Abra a página Propriedades do projeto.
2. Clique na página de propriedades Compilar.
3. Clique no botão Avançado.
4. Modifique a propriedade Endereço Básico de DLL.
Para definir programaticamente essa opção do compilador, confira BaseAddress.
Consulte também
ProcessModule.BaseAddress
Opções do compilador de C#
Gerenciando propriedades de solução e de projeto
-bugreport (opções do compilador C#)
23/10/2019 • 2 minutes to read • Edit Online
Especifica que as informações de depuração devem ser colocadas em um arquivo para análise posterior.
Sintaxe
-bugreport:file
Arguments
file
O nome do arquivo que conterá o relatório de bug.
Comentários
A opção -bugreport especifica que as informações a seguir devem ser colocadas em file :
Uma cópia de todos os arquivos de código-fonte na compilação.
Uma lista das opção do compilador usadas na compilação.
Informações de versão sobre o compilador, o tempo de execução e o sistema operacional.
Assemblies e módulos referenciados, salvos como dígitos hexadecimais, exceto os assemblies que vêm com
o .NET Framework e o SDK.
Saída do compilador, se houver.
Uma solicitação de descrição do problema.
Uma solicitação de como você acha que o problema deve ser corrigido.
Se essa opção for usada com -errorreport:prompt ou -errorreport:send, as informações no arquivo serão
enviadas à Microsoft Corporation.
Como uma cópia de todos os arquivos de código-fonte será colocada no file , talvez seja desejável reproduzir o
suposto defeito do código no programa mais curto possível.
Essa opção do compilador não está disponível no Visual Studio e não pode ser alterada programaticamente.
Observe que os conteúdos do arquivo gerado expõem o código-fonte, o que pode resultar na divulgação acidental
de informações.
Consulte também
Opções do compilador de C#
-errorreport (opções do compilador do C#)
Gerenciando propriedades de solução e de projeto
-checked (opções do compilador C#)
23/10/2019 • 2 minutes to read • Edit Online
A opção -checked especifica se uma instrução de aritmética de inteiros que resulta em um valor fora do intervalo
do tipo de dados e que não está no escopo de uma palavra-chave checked ou unchecked, causa uma exceção de
tempo de execução.
Sintaxe
-checked[+ | -]
Comentários
Uma instrução de aritmética de inteiros que está no escopo de uma palavra-chave checked ou unchecked não
está sujeita ao efeito da opção -checked.
Se uma instrução de aritmética de inteiros que não está no escopo de uma palavra-chave checked ou unchecked
resultar em um valor fora do intervalo do tipo de dados e -checked+ (ou -checked) for usado na compilação, a
instrução causará uma exceção no tempo de execução. Se -checked- for usado na compilação, a instrução não
causará uma exceção em tempo de execução.
O valor padrão para essa opção é -checked- . A verificação de estouro é desabilitada.
Às vezes, as ferramentas automatizadas que são usadas para compilar grandes aplicativos definem -checked
como +. Um cenário para uso de -checked- é substituir o padrão global da ferramenta especificando -checked-.
Para definir esta opção do compilador no ambiente de desenvolvimento do Visual Studio
1. Abra a página Propriedades do projeto. Para obter mais informações, consulte Página Build, Designer de
Projeto (C#).
2. Clique na página de propriedades Compilar.
3. Clique no botão Avançado.
4. Modifique a propriedade Procurar estouro/estouro negativo aritmético.
Para acessar programaticamente essa opção do compilador, confira CheckForOverflowUnderflow.
Exemplo
O comando a seguir compila t2.cs . O uso de -checked no comando especifica que qualquer instrução de
aritmética de inteiros no arquivo que não está no escopo de uma palavra-chave checked ou unchecked e que
resulta em um valor fora do intervalo do tipo de dados, causa uma exceção em tempo de execução.
Consulte também
Opções do compilador de C#
Gerenciando propriedades de solução e de projeto
-codepage (opções do compilador C#)
23/10/2019 • 2 minutes to read • Edit Online
Esta opção especifica qual página de código deve ser usada durante a compilação caso a página necessária não
seja a página de código padrão atual do sistema.
Sintaxe
-codepage:id
Arguments
id
Especifica a ID da página de código a ser usada para todos os arquivos de código-fonte na compilação.
Comentários
O compilador tentará primeiro interpretar todos os arquivos de origem como UTF -8. Se seus arquivos de código-
fonte estiverem em uma codificação diferente de UTF -8 e usarem caracteres diferentes de caracteres ASCII de 7
bits, use a opção -codepage para especificar qual página de código deve ser usada. -codepage se aplica a todos
os arquivos de código-fonte na compilação.
Consulte GetCPInfo para obter informações sobre como localizar quais páginas de código têm suporte do
sistema.
Essa opção do compilador não está disponível no Visual Studio e não pode ser alterada programaticamente.
Consulte também
Opções do compilador de C#
Gerenciando propriedades de solução e de projeto
-debug (opções do compilador C#)
23/10/2019 • 3 minutes to read • Edit Online
A opção -debug faz o compilador gerar informações de depuração e colocá-las nos arquivos de saída.
Sintaxe
-debug[+ | -]
-debug:{full | pdbonly}
Arguments
+ | -
Especificar + ou apenas -debug faz o compilador gerar informações de depuração e colocá-las em um de banco
de dados de programa (arquivo .pdb). Especificar - , que será aplicado se -debug não for especificado, faz com
que nenhuma informação de depuração seja criada.
full | pdbonly
Especifica o tipo de informações de depuração geradas pelo compilador. O argumento completo, que será
aplicado se -debug:pdbonly não for especificado, permite anexar um depurador ao programa em execução.
Especificar pdbonly habilita a depuração do código-fonte quando o programa é iniciado no depurador, mas exibirá
somente o assembler quando o programa em execução for anexado ao depurador.
Comentários
Use essa opção para criar builds de depuração. Caso -debug, -debug+ ou -debug:full não sejam especificados,
não será possível depurar o arquivo de saída do programa.
Caso -debug:full seja usado, lembre-se de que isso influenciará a velocidade e o tamanho do código otimizado
JIT e haverá um pequeno impacto na qualidade do código com -debug:full. Recomenda-se -debug:pdbonly ou
nenhum PDB para gerar código de versão.
NOTE
Uma diferença entre -debug:pdbonly e -debug:full é que, com -debug:full, o compilador emite uma
DebuggableAttribute, que é usada para avisar ao compilador JIT que as informações de depuração estão disponíveis.
Portanto, haverá um erro se o código contiver DebuggableAttribute definido como false, caso -debug:full seja usado.
Para obter informações sobre como configurar o desempenho de depuração de um aplicativo, consulte
Facilitando a Depuração de uma Imagem.
Para alterar o local do arquivo .pdb, consulte -pdb (Opções do Compilador C#).
Para definir esta opção do compilador no ambiente de desenvolvimento do Visual Studio
1. Abra a página Propriedades do projeto.
2. Clique na página de propriedades Compilar.
3. Clique no botão Avançado.
4. Modifique a propriedade Informações de Depuração.
Para saber mais sobre como definir essa opção do compilador programaticamente, veja DebugSymbols.
Exemplo
Coloque as informações de depuração no arquivo de saída app.pdb :
Consulte também
Opções do compilador de C#
Gerenciando propriedades de solução e de projeto
-define (opções do compilador C#)
23/10/2019 • 3 minutes to read • Edit Online
A opção -define define name como um símbolo em todos os arquivos de código-fonte do seu programa.
Sintaxe
-define:name[;name2]
Arguments
name, name2
O nome de um ou mais símbolos que você deseja definir.
Comentários
A opção -define tem o mesmo efeito que usar uma diretiva de pré-processador #define, exceto que a opção do
compilador está em vigor para todos os arquivos no projeto. Um símbolo permanece definido em um arquivo de
origem até que uma diretiva #undef remova a definição no arquivo de origem. Quando você usa a opção -define,
uma diretiva #undef em um arquivo não terá nenhum efeito em outros arquivos de código-fonte no projeto.
Você pode usar os símbolos criados por essa opção com #if, #else, #elif e #endif para compilar os arquivos de
origem condicionalmente.
-d é a forma abreviada de -define.
Você pode definir vários símbolos com -define, usando um ponto e vírgula ou uma vírgula para separar os
nomes dos símbolos. Por exemplo:
-define:DEBUG;TUESDAY
O compilador do C# não define símbolos ou macros que podem ser usados em seu código-fonte. Todas as
definições de símbolo devem ser definidas pelo usuário.
NOTE
O #define do C# não permite que um valor seja dado a um símbolo, como nas linguagens como a C++. Por exemplo,
#define não pode ser usado para criar uma macro ou para definir uma constante. Se você precisar definir uma constante,
use uma variável enum . Se você quiser criar uma macro de estilo C++, considere alternativas como os genéricos. Como as
macros são notoriamente propensas a erros, o C# não permite o uso delas, mas oferece alternativas mais seguras.
Consulte também
Opções do compilador de C#
Gerenciando propriedades de solução e de projeto
-delaysign (opções do compilador C#)
23/10/2019 • 2 minutes to read • Edit Online
Essa opção faz com que o compilador reserve espaço no arquivo de saída para que uma assinatura digital possa
ser adicionada mais tarde.
Sintaxe
-delaysign[ + | - ]
Arguments
+ | -
Use -delaysign- se você quiser um assembly totalmente assinado. Use -delaysign+ se você apenas desejar
colocar a chave pública no assembly. O padrão é -delaysign- .
Comentários
A opção -delaysign não tem nenhum efeito a menos que seja usada com -keyfile ou -keycontainer.
As opções -delaysign e -publicsign são mutuamente exclusivas.
Quando você solicita um assembly totalmente assinado, o compilador usa o hash no arquivo que contém o
manifesto (metadados de assembly) e sinaliza esse hash com a chave particular. Essa operação cria uma
assinatura digital que é armazenada no arquivo que contém o manifesto. Quando um assembly é assinado com
atraso, o compilador não calcula e armazena a assinatura, mas reserva o espaço no arquivo, de forma que a
assinatura possa ser adicionada depois.
Por exemplo, o uso de -delaysign+ permite que um testador coloque o assembly no cache global. Após o teste, é
possível assinar completamente o assembly, colocando a chave particular no assembly com o utilitário Assembly
Linker.
Para obter mais informações, consulte Criando e usando assemblies de nomes fortes e Atraso na Assinatura de
um Assembly.
Para definir esta opção do compilador no ambiente de desenvolvimento do Visual Studio
1. Abra a página Propriedades do projeto.
2. Modifique a propriedade Apenas adiar a assinatura.
Para saber mais sobre como definir essa opção do compilador programaticamente, veja DelaySign.
Consulte também
Opção -publicsign do C#
Opções do compilador de C#
Gerenciando propriedades de solução e de projeto
-deterministic
04/11/2019 • 2 minutes to read • Edit Online
Faz com que o compilador produza um assembly cuja saída byte a byte é idêntica entre compilações para
entradas idênticas.
Sintaxe
-deterministic
Comentários
Por padrão, a saída do compilador de um determinado conjunto de entradas é exclusiva, uma vez que o
compilador adiciona um carimbo de data/hora e um GUID gerado com base em números aleatórios. Use a opção
-deterministic para produzir um assembly determinística, cujo conteúdo binário seja idêntico entre compilações,
desde que a entrada permaneça a mesma.
O compilador considera as seguintes entradas com a finalidade de determinismo:
A sequência de parâmetros de linha de comando.
O conteúdo do arquivo de resposta .rsp do compilador.
A versão precisa do compilador usado e seus assemblies referenciados.
O caminho do diretório atual.
O conteúdo binário de todos os arquivos passados explicitamente para o compilador direta ou indiretamente,
incluindo:
Arquivos de origem
Assemblies referenciados
Módulos referenciados
Recursos
O arquivo de chave de nome forte
@ arquivos de resposta
Analisadores
Conjuntos de regras
Arquivos adicionais que podem ser usados por analisadores
A cultura atual (para o idioma no qual as mensagens de diagnóstico e exceção são produzidas).
A codificação padrão (ou a página de código atual) se a codificação não for especificada.
A existência, a inexistência e o conteúdo dos arquivos em caminhos de pesquisa do compilador (especificados,
por exemplo, por -lib ou -recurse ).
A plataforma CLR na qual o compilador é executado.
O valor de %LIBPATH% , que pode afetar o carregamento de dependência do analisador.
Quando as fontes estão disponíveis publicamente, a compilação determinística pode ser usada para estabelecer se
um binário é compilado de uma fonte confiável. Isso também pode ser útil em um sistema de compilação
contínua para determinar se as etapas de compilação dependentes de alterações para um binário precisam ser
executadas.
Consulte também
Opções do compilador de C#
Gerenciando propriedades de solução e de projeto
-doc (opções do compilador C#)
04/11/2019 • 2 minutes to read • Edit Online
Sintaxe
-doc:file
Arguments
file
O arquivo de saída para XML, que é populado com os comentários nos arquivos de código-fonte da
compilação.
Comentários
Nos arquivos de código-fonte, os comentários de documentação que precedem o seguinte podem ser
processados e adicionados ao arquivo XML:
Tipos definidos pelo usuário, como classe, delegado ou interface
Membros como um campo, evento, propriedade ou método
O arquivo de código-fonte que contém Main é gerado primeiro no XML.
Para usar o arquivo .xml gerado para uso com o recurso IntelliSense, deixe o nome do arquivo .xml
igual ao do assembly a que você deseja dar suporte e, em seguida, verifique se o arquivo .xml está no
mesmo diretório que o assembly. Sendo assim, quando o assembly for referenciado no projeto do
Visual Studio, o arquivo .xml também será encontrado. Consulte Fornecendo comentários de código
para obter mais informações.
A menos que você compile com -target:module, file conterá marcas <assembly></assembly>
especificando o nome do arquivo que contém o manifesto do assembly para o arquivo de saída da
compilação.
NOTE
A opção -doc se aplica a todos os arquivos de entrada ou, se definida nas Configurações do Projeto, todos os
arquivos no projeto. Para desabilitar avisos relacionados aos comentários de documentação para um arquivo ou
uma seção específica do código, use #pragma warning.
Consulte Marcas recomendadas para comentários de documentação para ver maneiras de gerar a
documentação dos comentários em seu código.
Para definir esta opção do compilador no ambiente de desenvolvimento do Visual Studio
1. Abra a página Propriedades do projeto.
2. Clique na guia Build.
3. Modifique a propriedade Arquivo de documentação XML.
Para saber mais sobre como definir essa opção do compilador programaticamente, veja
DocumentationFile.
Consulte também
Opções do compilador de C#
Gerenciando propriedades de solução e de projeto
-errorreport (opções do compilador C#)
23/10/2019 • 3 minutes to read • Edit Online
Esta opção fornece uma maneira conveniente de relatar um erro interno do compilador do C# à Microsoft.
NOTE
No Windows Vista e Windows Server 2008, as configurações de relatório de erros que você faz para o Visual Studio não
substituem as configurações feitas pelo WER (Relatório de Erros do Windows). As configurações do WER sempre têm
precedência sobre as configurações de relatório de erros do Visual Studio.
Sintaxe
-errorreport:{ none | prompt | queue | send }
Arguments
none
Relatórios sobre erros internos do compilador não serão coletados ou enviados à Microsoft.
aviso Solicita que você envie um relatório quando recebe um erro de compilador interno. prompt é o padrão
quando você compila um aplicativo no ambiente de desenvolvimento.
fila Enfileira o relatório de erros. Quando você faz logon com credenciais administrativas, pode relatar falhas
desde a última vez que você fez logon. Não será solicitado que você envie relatórios de falhas mais de uma vez a
cada três dias. queue é o padrão quando você compila um aplicativo na linha de comando.
Enviar Envia automaticamente relatórios de erros de compilador interno à Microsoft. Para habilitar essa opção,
primeiro você deve concordar com a política de coleta de dados da Microsoft. Na primeira vez que você
especificar -errorreport:send em um computador, uma mensagem de compilador indicará um site que contém a
política de coleta de dados da Microsoft.
Comentários
Um ICE (erro interno do compilador) ocorre quando o compilador não pode processar um arquivo de código-
fonte. Quando um ICE ocorre, o compilador não produz um arquivo de saída ou qualquer diagnóstico útil que
você pode usar para corrigir o código.
Em versões anteriores, quando você recebia um ICE, era incentivado a entrar em contato com o Microsoft Product
Support Services para relatar o problema. Usando -errorreport, você pode fornecer informações do ICE à equipe
do Visual C#. Os relatórios de erro podem ajudar a melhorar versões futuras do compilador.
A capacidade do usuário de enviar relatórios depende das permissões de política de usuário e de computador.
Para definir esta opção do compilador no ambiente de desenvolvimento do Visual Studio
1. Abra a página Propriedades do projeto. Para obter mais informações, consulte Página Build, Designer de
Projeto (C#).
2. Clique na página de propriedades Compilar.
3. Clique no botão Avançado.
4. Modifique a propriedade Relatório de Erros do Compilador Interno.
Para saber mais sobre como definir essa opção do compilador programaticamente, veja ErrorReport.
Consulte também
Opções do compilador de C#
-filealign (opções do compilador C#)
23/10/2019 • 2 minutes to read • Edit Online
A opção -filealign permite que você especifique o tamanho das seções em seu arquivo de saída.
Sintaxe
-filealign:number
Arguments
number
Um valor que especifica o tamanho das seções no arquivo de saída. Os valores válidos são 512, 1024, 2048, 4096
e 8192. Esses valores estão em bytes.
Comentários
Cada seção será alinhada em um limite que é um múltiplo do valor -filealign. Não há nenhum padrão fixo. Se -
filealign não é especificada, o Common Language Runtime escolhe um padrão em tempo de compilação.
Ao especificar o tamanho da seção, você afeta o tamanho do arquivo de saída. Modificar o tamanho da seção pode
ser útil para programas que serão executados em dispositivos menores.
Use DUMPBIN para ver informações sobre as seções em seu arquivo de saída.
Para definir esta opção do compilador no ambiente de desenvolvimento do Visual Studio
1. Abra a página Propriedades do projeto.
2. Clique na página de propriedades Compilar.
3. Clique no botão Avançado.
4. Modifique a propriedade Alinhamento de Arquivo.
Para saber mais sobre como definir essa opção do compilador programaticamente, veja FileAlignment.
Consulte também
Opções do compilador de C#
Gerenciando propriedades de solução e de projeto
-fullpaths (opções do compilador C#)
23/10/2019 • 2 minutes to read • Edit Online
A opção -fullpaths faz com que o compilador especifique o caminho completo para o arquivo ao listar erros de
compilação e avisos.
Sintaxe
-fullpaths
Comentários
Por padrão, erros e avisos oriundos da compilação especificam o nome do arquivo no qual o erro foi encontrado.
A opção -fullpaths faz com que o compilador especifique o caminho completo para o arquivo.
Essa opção do compilador não está disponível no Visual Studio e não pode ser alterada programaticamente.
Consulte também
Opções do compilador de C#
/help, /? (Opções do Compilador C#)
23/10/2019 • 2 minutes to read • Edit Online
Essa opção envia uma lista de opções do compilador e uma breve descrição de cada opção para stdout.
Sintaxe
-help
-?
Comentários
Se essa opção for incluída em uma compilação, nenhum arquivo de saída será criado e a compilação não ocorrerá.
Essa opção do compilador não está disponível no Visual Studio e não pode ser alterada programaticamente.
Consulte também
Opções do compilador de C#
Gerenciando propriedades de solução e de projeto
-highentropyva (opções do compilador C#)
23/10/2019 • 2 minutes to read • Edit Online
Sintaxe
-highentropyva[+ | -]
Arguments
+ | -
Essa opção especifica que um executável de 64 bits ou um executável que está marcado com a opção do
compilador -platform:anycpu é compatível com um espaço de endereço virtual de alta entropia. A opção está
desabilitada por padrão. Use -highentropyva+ ou -highentropyva para habilitá-la.
Comentários
A opção -highentropyva permite que as versões compatíveis do kernel do Windows usem níveis mais altos de
entropia ao randomizar o layout do espaço de endereço de um processo como parte da ASLR. O uso de níveis
mais altos de entropia significa que um número maior de endereços pode ser alocado para regiões de memória
como pilhas e heaps. Como resultado, é mais difícil adivinhar a localização de uma região específica da memória.
Quando a opção do compilador -highentropyva é especificada, o executável de destino e todos os módulos dos
quais ele depende devem ser capazes de manipular valores de ponteiro que sejam maiores que 4 GB (gigabytes)
quando eles estiverem em execução como um processo de 64 bits.
-keycontainer (opções do compilador C#)
23/10/2019 • 2 minutes to read • Edit Online
Sintaxe
-keycontainer:string
Arguments
string
O nome do contêiner de chave de nome forte.
Comentários
Quando a opção -keycontainer for usada, o compilador criará um componente compartilhável. O compilador
insere uma chave pública do contêiner especificado no manifesto do assembly e assina o assembly final com a
chave privada. Para gerar um arquivo de chave, digite sn -k file na linha de comando. sn -i instala o par de
chaves no contêiner. Não há suporte para essa opção quando o compilador é executado no CoreCLR. Para
assinar um assembly ao compilar no CoreCLR, use a opção -keyfile.
Se você compilar com -target:module, o nome do arquivo de chave será mantido no módulo e incorporado no
assembly quando você compilar esse módulo em um assembly com -addmodule.
Também é possível especificar essa opção como um atributo personalizado
(System.Reflection.AssemblyKeyNameAttribute) no código-fonte de qualquer módulo MSIL (Microsoft
Intermediate Language).
Também é possível passar suas informações de criptografia para o compilador com -keyfile. Use -delaysign se
você quiser que a chave pública seja adicionada ao manifesto do assembly, mas deseja atrasar a assinatura do
assembly até que ela seja testada.
Para obter mais informações, consulte Criando e usando assemblies de nomes fortes e Atraso na Assinatura de
um Assembly.
Para definir esta opção do compilador no ambiente de desenvolvimento do Visual Studio
1. Essa opção do compilador não está disponível no ambiente de desenvolvimento do Visual Studio.
Você pode acessar programaticamente essa opção do compilador com AssemblyKeyContainerName.
Consulte também
Opção -keyfile do compilador C#
Opções do compilador de C#
Gerenciando propriedades de solução e de projeto
-keyfile (opções do compilador C#)
23/10/2019 • 3 minutes to read • Edit Online
Sintaxe
-keyfile:file
Arguments
TERMO DEFINIÇÃO
Comentários
Quando esta opção é usada, o compilador insere a chave pública da linha especificada no manifesto do
assembly e, em seguida, assina o assembly definitivo com a chave privada. Para gerar um arquivo de chave,
digite sn -k file na linha de comando.
Se você compilar com -target:module, o nome do arquivo de chave será mantido no módulo e incorporado no
assembly, criado quando você compila um assembly com -addmodule.
Também é possível passar suas informações de criptografia para o compilador com -keycontainer. Use -
delaysign se quiser um assembly parcialmente assinado.
Caso -keyfile e -keycontainer sejam especificados (pela opção de linha de comando ou pelo atributo
personalizado) na mesma compilação, o compilador tentará primeiro o contêiner de chaves. Se isso ocorrer, o
assembly será assinado com as informações no contêiner de chaves. Se o compilador não localizar o contêiner
de chaves, ele tentará o arquivo especificado com -keyfile. Se isso ocorrer, o assembly será assinado com as
informações no arquivo de chave e as informações da chave serão instaladas no contêiner de chaves
(semelhante a sn -i), de forma que, na próxima compilação, o contêiner de chaves será válido.
Observe que um arquivo de chave pode conter somente a chave pública.
Para obter mais informações, consulte Criando e usando assemblies de nomes fortes e Atraso na Assinatura de
um Assembly.
Para definir esta opção do compilador no ambiente de desenvolvimento do Visual Studio
1. Abra a página Propriedades do projeto.
2. Clique na página de propriedades Assinatura.
3. Modifique a propriedade Escolha um arquivo de chave de nome forte.
Você pode acessar programaticamente essa opção do compilador com AssemblyOriginatorKeyFile.
Consulte também
Opções do compilador de C#
Gerenciando propriedades de solução e de projeto
-langversion (opções do compilador C#)
04/11/2019 • 7 minutes to read • Edit Online
Faz com que o compilador aceite somente a sintaxe incluída na especificação de linguagem C# escolhida.
Sintaxe
-langversion:option
Arguments
option
Os seguintes valores são válidos:
OPÇÃO SIGNIFICADO
A versão da linguagem padrão depende da estrutura de destino do aplicativo e da versão do SDK ou do Visual
Studio instalado. Essas regras são definidas no artigo Configurando a versão do idioma .
Comentários
Os metadados referenciados pelo seu aplicativo de C# não estão sujeitos à opção do compilador -langversion.
Como cada versão do compilador do C# contém extensões para a especificação de linguagem, -langversion não
dá a funcionalidade equivalente de uma versão anterior do compilador.
Além disso, embora as atualizações de versão do C# geralmente coincidam com as versões principais do .NET
Framework, a nova sintaxe e as funcionalidades não estão necessariamente vinculadas a essa versão de estrutura
específica. Embora as novas funcionalidades definitivamente exijam uma nova atualização do compilador que
também é liberada junto com a revisão do C#, cada funcionalidade específica tem seus próprios requisitos
mínimos de API ou do Common Language Runtime do .NET que podem permitir que ela seja executada em
estruturas de nível inferior com a inclusão de pacotes NuGet ou de outras bibliotecas.
Independentemente de qual configuração -langversion for usada, você usará a versão atual do Common
Language Runtime para criar seu .exe ou .dll. Uma exceção são os assemblies amigáveis e -moduleassemblyname
(Opção do compilador do C#), que funcionarão em -langversion:ISO -1.
Para outras maneiras de especificar a C# versão do idioma, consulte o artigo selecionar a versão C# do idioma .
Para saber mais sobre como definir essa opção do compilador programaticamente, veja LanguageVersion.
Especificação da linguagem C#
VERSION LINK DESCRIÇÃO
Consulte também
Opções do compilador de C#
Gerenciando propriedades de solução e de projeto
-lib (opções do compilador C#)
23/10/2019 • 3 minutes to read • Edit Online
A opção -lib especifica o local dos assemblies referenciados por meio da opção -reference (Opções do
Compilador do C#).
Sintaxe
-lib:dir1[,dir2]
Arguments
dir1
Um diretório para o compilador examinar se um assembly referenciado não foi encontrado no diretório de
trabalho atual (o diretório do qual você está invocando o compilador) ou no diretório de sistema do Common
Language Runtime.
dir2
Um ou mais diretórios adicionais a serem pesquisados para referências de assembly. Separe os nomes de
diretório adicionais com uma vírgula e não inclua espaços em branco entre eles.
Comentários
O compilador pesquisa referências de assembly que não são totalmente qualificadas na seguinte ordem:
1. Diretório de trabalho atual. Esse é o diretório do qual o compilador é invocado.
2. O diretório de sistema do Common Language Runtime.
3. Diretórios especificados por -lib.
4. Diretórios especificados pela variável de ambiente LIB.
Use -reference para especificar uma referência de assembly.
-lib é aditivo; especificá-lo mais de uma vez acrescenta a valores anteriores.
Uma alternativa ao uso de -lib é copiar todos assemblies necessários para o diretório de trabalho; isso permitirá
que você simplesmente passe o nome do assembly para -reference. Em seguida, é possível excluir os assemblies
do diretório de trabalho. Como o caminho para o assembly dependente não foi especificado no manifesto do
assembly, o aplicativo pode ser iniciado no computador de destino e localizará e usará o assembly no cache de
assembly global.
O fato de o compilador poder referenciar o assembly não implica que o Common Language Runtime será capaz
de localizar e carregar o assembly em tempo de execução. Consulte Como o tempo de execução localiza
assemblies para obter detalhes sobre como o runtime pesquisa assemblies referenciados.
Para definir esta opção do compilador no ambiente de desenvolvimento do Visual Studio
1. Abra a caixa de diálogo Páginas de Propriedades do projeto.
2. Clique na página de propriedades Caminho de Referências.
3. Modifique o conteúdo da caixa de listagem.
Para saber mais sobre como definir essa opção do compilador programaticamente, veja ReferencePath.
Exemplo
Compile t2.cs para criar um arquivo .exe. O compilador examinará referências de assembly no diretório de
trabalho e no diretório raiz da unidade C.
Consulte também
Opções do compilador de C#
Gerenciando propriedades de solução e de projeto
-link (opções do compilador C#)
23/10/2019 • 7 minutes to read • Edit Online
Faz com que o compilador disponibilize as informações de tipo COM nos assemblies especificados para o
projeto sendo compilado no momento.
Sintaxe
-link:fileList
// -or-
-l:fileList
Arguments
fileList
Necessário. Lista delimitada por vírgulas de nomes de arquivo do assembly. Se o nome do arquivo contém um
espaço, coloque o nome entre aspas.
Comentários
A opção -link permite que você implante um aplicativo que inseriu informações de tipo. O aplicativo pode
usar tipos em um assembly de tempo de execução que implementa as informações de tipo inseridas sem a
necessidade de uma referência ao assembly de tempo de execução. Se forem publicadas várias versões do
assembly de tempo de execução, o aplicativo que contém as informações de tipo inseridas poderá trabalhar com
as várias versões sem precisar ser recompilado. Para obter um exemplo, confira Passo a passo: inserir tipos de
assemblies gerenciados.
Usar a opção -link é especialmente útil quando você está trabalhando com a interoperabilidade COM. Você
pode inserir tipos COM para que seu aplicativo não precise mais de um PIA (assembly de interoperabilidade
primário) no computador de destino. A opção -link instrui o compilador a inserir as informações de tipo de
COM do assembly de interoperabilidade referenciado no código compilado resultante. O tipo COM é
identificado pelo valor CLSID (GUID ). Como resultado, o aplicativo pode ser executado em um computador de
destino que tem os mesmos tipos COM instalados com os mesmos valores CLSID. Os aplicativos que
automatizam o Microsoft Office são um bom exemplo. Como aplicativos como o Office normalmente mantêm o
mesmo valor CLSID entre diferentes versões, seu aplicativo pode usar os tipos COM referenciados contanto que
o .NET Framework 4 ou posterior esteja instalado no computador de destino e seu aplicativo use métodos,
propriedades ou eventos que estão incluídos nos tipos COM referenciados.
A opção -link incorpora apenas interfaces, estruturas e delegados. Não há suporte para a inserção de classes
COM.
NOTE
Quando você cria uma instância de um tipo COM inserido no seu código, você deve criar a instância usando a interface
apropriada. Tentar criar uma instância de um tipo COM inserido usando o CoClass causa um erro.
Para definir a opção -linkem Visual Studio, adicione uma referência de assembly e defina a propriedade
Embed Interop Types como true. O valor padrão da propriedade Embed Interop Types é false.
Se você vincular a um assembly COM (Assembly A) que em si faz referência a outro assembly COM (Assembly
B ), também precisará vincular ao Assembly B se uma das seguintes opções for verdadeira:
Um tipo do Assembly A herda de um tipo ou implementa uma interface do Assembly B.
Um campo, propriedade, evento ou método que tem um tipo de retorno ou de parâmetro do Assembly B
é invocado.
Como a opção do compilador -reference, a opção do compilador -link usa o arquivo de resposta Csc.rsp, que
faz referência a assemblies do .NET Framework usados com frequência. Use a opção do compilador -noconfig
se não quiser que o compilador use o arquivo Csc.rsp.
A forma abreviada de -link é -l .
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using Microsoft.Office.Interop.Excel;
}
}
No exemplo a seguir, o código do cliente pode chamar o método que retorna a interface genérica IList sem
erros.
public class Client
{
public void Main()
{
Utility util = new Utility();
Exemplo
O código a seguir compila o arquivo de origem OfficeApp.cs e faz referência aos assemblies de COMData1.dll e
COMData2.dll para produzir OfficeApp.exe .
Consulte também
Opções do compilador de C#
Passo a passo: inserindo tipos de assemblies gerenciados
-reference (opções do compilador do C#)
-noconfig (opções do compilador do C#)
Build pela linha de comando com csc.exe
Visão geral sobre interoperabilidade
-linkresource (opções do compilador C#)
23/10/2019 • 3 minutes to read • Edit Online
Cria um link para um recurso do .NET Framework no arquivo de saída. O arquivo de recurso não é adicionado ao
arquivo de saída. Isso é diferente da opção -resource que insere um arquivo de recurso no arquivo de saída.
Sintaxe
-linkresource:filename[,identifier[,accessibility-modifier]]
Arguments
filename
O arquivo de recurso do .NET Framework ao qual você deseja vincular do assembly.
identifier (opcional)
O nome lógico do recurso; o nome usado para carregar o recurso. O padrão é o nome do arquivo.
accessibility-modifier (opcional)
A acessibilidade do recurso: público ou privado. O padrão é público.
Comentários
Por padrão, os recursos vinculados são públicos no assembly quando são criados usando o compilador C#. Para
tornar os recursos privados, especifique private como o modificador de acessibilidade. Não é permitido nenhum
outro modificador diferente de public ou de private .
-linkresource requer uma das opções -target diferentes de -target:module.
Se filename for um arquivo de recurso do .NET Framework criado, por exemplo, por Resgen.exe ou no ambiente
de desenvolvimento, ele poderá ser acessado com membros no namespace System.Resources. Para obter mais
informações, consulte System.Resources.ResourceManager. Para todos os outros recursos, use os métodos
GetManifestResource na classe Assembly para acessar o recurso em tempo de execução.
O arquivo especificado em filename pode ter qualquer formato. Por exemplo, crie uma parte DLL nativa do
assembly de maneira que possa ser instalada no cache de assembly global e acessado no código gerenciado no
assembly. O segundo dos exemplos a seguir mostra como fazer isso. É possível fazer a mesma coisa no Assembly
Linker. O terceiro dos exemplos a seguir mostra como fazer isso. Para obter mais informações, consulte Al.exe
(Assembly Linker) e Trabalhando com assemblies e o cache de assembly global.
-linkres é a forma abreviada de -linkresource.
Essa opção do compilador não está disponível no Visual Studio e não pode ser alterada programaticamente.
Exemplo
Compile in.cs e vincule ao arquivo de recurso rf.resource :
Exemplo
Este exemplo faz a mesma coisa que o anterior, mas usando as opções do Assembly Linker.
Consulte também
Opções do compilador de C#
Al.exe (Assembly Linker)
Como trabalhar com assemblies e o cache de assembly global
Gerenciando propriedades de solução e de projeto
-main (opções do compilador C#)
23/10/2019 • 2 minutes to read • Edit Online
Esta opção especifica a classe que contém o ponto de entrada para o programa, se mais de uma classe contiver
um método Main.
Sintaxe
-main:class
Arguments
class
O tipo que contém o método Main.
O nome de classe informado deve ser totalmente qualificado. Ele deve incluir o namespace completo que contém
a classe, seguido do nome de classe. Por exemplo, quando o método Main é localizado dentro da classe Program
no namespace MyApplication.Core , a opção do compilador deve ser -main:MyApplication.Core.Program .
Comentários
Se sua compilação incluir mais de um tipo com um método Main, você poderá especificar qual tipo contém o
método Main que deseja usar como o ponto de entrada para o programa.
Essa opção é para uso durante a compilação de um arquivo .exe.
Para definir esta opção do compilador no ambiente de desenvolvimento do Visual Studio
1. Abra a página Propriedades do projeto.
2. Clique na página de propriedades do Aplicativo.
3. Modifique a propriedade Objeto de inicialização.
Para definir programaticamente essa opção do compilador, confira StartupObject.
Exemplo
Compile t2.cs e t3.cs , especificando que o método Main será encontrado em Test2 :
Consulte também
Opções do compilador de C#
Gerenciando propriedades de solução e de projeto
-moduleassemblyname (opção do compilador C#)
23/10/2019 • 2 minutes to read • Edit Online
Especifica um assembly cujos tipos não públicos podem ser acessados por um .netmodule.
Sintaxe
-moduleassemblyname:assembly_name
Arguments
assembly_name
O nome do assembly cujos tipos não públicos podem ser acessados por um .netmodule.
Comentários
O -moduleassemblyname deverá ser usado ao compilar um .netmodule e quando as condições a seguir forem
true:
O .netmodule precisa acessar tipos não públicos em um assembly existente.
Você sabe o nome do assembly no qual o .netmodule será compilado.
O assembly existente concedeu acesso de assembly amigável ao assembly no qual o .netmodule será
compilado.
Para obter mais informações sobre a compilação de um .netmodule, consulte -target:module (Opções do
compilador do C#).
Para obter mais informações sobre assemblies amigos, consulte Assemblies Amigáveis.
Essa opção não está disponível de dentro do ambiente de desenvolvimento; ela só está disponível quando se
compila na linha de comando.
Essa opção do compilador não está disponível no Visual Studio e não pode ser alterada programaticamente.
Exemplo
Este exemplo compila um assembly com um tipo privado que concede acesso de assembly amigável a um
assembly denominado csman_an_assembly.
// moduleassemblyname_1.cs
// compile with: -target:library
using System;
using System.Runtime.CompilerServices;
[assembly:InternalsVisibleTo ("csman_an_assembly")]
class An_Internal_Class
{
public void Test()
{
Console.WriteLine("An_Internal_Class.Test called");
}
}
Exemplo
Este exemplo compila um .netmodule que acessa um tipo não público no assembly moduleassemblyname_1.dll.
Sabendo que esse .netmodule será compilado dentro de um assembly chamado csman_an_assembly, é possível
especificar -moduleassemblyname, permitindo que o .netmodule acesse tipos não públicos em um assembly
que tenha concedido acesso de assembly amigável ao csman_an_assembly.
// moduleassemblyname_2.cs
// compile with: -moduleassemblyname:csman_an_assembly -target:module -reference:moduleassemblyname_1.dll
class B {
public void Test() {
An_Internal_Class x = new An_Internal_Class();
x.Test();
}
}
Exemplo
Este exemplo de código compila o assembly csman_an_assembly, referenciando o assembly compilado
anteriormente e o .netmodule.
// csman_an_assembly.cs
// compile with: -addmodule:moduleassemblyname_2.netmodule -reference:moduleassemblyname_1.dll
class A {
public static void Main() {
B bb = new B();
bb.Test();
}
}
Consulte também
Opções do compilador de C#
Gerenciando propriedades de solução e de projeto
-noconfig (opções do compilador C#)
23/10/2019 • 2 minutes to read • Edit Online
A opção -noconfig informa que o compilador não deve compilar com o arquivo csc.rsp, localizado no mesmo
diretório que o arquivo csc.exe e carregado dele.
Sintaxe
-noconfig
Comentários
O arquivo csc.rsp referencia todos os assemblies que acompanham o .NET Framework. As referências reais que
o ambiente de desenvolvimento do Visual Studio .NET inclui dependem do tipo de projeto.
É possível modificar o arquivo csc.rsp e especificar opções do compilador adicionais que devem ser incluídas em
cada compilação na linha de comando com csc.exe (exceto a opção -noconfig).
O compilador processa as opções passadas para o comando csc pela última vez. Portanto, qualquer opção na
linha de comando substitui a configuração da mesma opção no arquivo csc.rsp.
Se você não desejar que o compilador procure e use as configurações no arquivo csc.rsp, especifique -noconfig.
Essa opção do compilador não está disponível no Visual Studio e não pode ser alterada programaticamente.
Consulte também
Opções do compilador de C#
Gerenciando propriedades de solução e de projeto
-nologo (opções do compilador C#)
23/10/2019 • 2 minutes to read • Edit Online
A opção -nologo inibe a exibição da faixa de conexão quando o compilador é iniciado e inibe a exibição de
mensagens informativas durante a compilação.
Sintaxe
-nologo
Comentários
Essa opção não está disponível de dentro do ambiente de desenvolvimento; ela só está disponível quando se
compila na linha de comando.
Essa opção do compilador não está disponível no Visual Studio e não pode ser alterada programaticamente.
Consulte também
Opções do compilador de C#
Gerenciando propriedades de solução e de projeto
-nostdlib (opções do compilador C#)
23/10/2019 • 2 minutes to read • Edit Online
Sintaxe
-nostdlib[+ | -]
Comentários
Use essa opção se desejar definir ou criar seus próprios objetos e namespace System.
Se você não especificar -nostdlib, o mscorlib.dll será importado no programa (o mesmo que especificar -
nostdlib- ). Especificar -nostdlib é o mesmo que especificar -nostdlib+ .
Para definir essa opção do compilador no Visual Studio
NOTE
As instruções a seguir se aplicam apenas ao Visual Studio 2015 (e a versões anteriores). A propriedade de build Não
referenciar mscorlib.dll não existe no Visual Studio 2017.
Consulte também
Opções do compilador de C#
-nowarn (opções do compilador C#)
23/10/2019 • 2 minutes to read • Edit Online
A opção -nowarn permite suprimir a exibição de um ou mais avisos pelo compilador. Separe vários números de
aviso com uma vírgula.
Sintaxe
-nowarn:number1[,number2,...]
Arguments
, number2
number1
Números de aviso que você deseja que o compilador suprima.
Comentários
Você só precisa especificar a parte numérica do identificador de aviso. Por exemplo, se quiser suprimir CS0028,
você pode especificar -nowarn:28 .
O compilador ignorará silenciosamente números de aviso passados para -nowarn que eram válidos em versões
anteriores, mas que foram removidos do compilador. Por exemplo, CS0679 era válido no compilador no Visual
Studio .NET 2002, mas foi removido posteriormente.
Os avisos a seguir não podem ser suprimidos pela opção -nowarn :
Aviso do compilador (nível 1) CS2002
Aviso do compilador (nível 1) CS2023
Aviso do compilador (nível 1) CS2029
Para definir esta opção do compilador no ambiente de desenvolvimento do Visual Studio
1. Abra a página Propriedades do projeto. Para obter detalhes, consulte Página de Build, Designer de Projeto
(C#).
2. Clique na página de propriedades Compilar.
3. Modifique a propriedade Suprimir Avisos.
Para saber mais sobre como definir essa opção do compilador programaticamente, veja DelaySign.
Consulte também
Opções do compilador de C#
Gerenciando propriedades de solução e de projeto
Erros do Compilador do C#
-nowin32manifest (opções do compilador C#)
23/10/2019 • 2 minutes to read • Edit Online
Use a opção -nowin32manifest para instruir o compilador a não inserir nenhum manifesto do aplicativo no
arquivo executável.
Sintaxe
-nowin32manifest
Comentários
Quando essa opção for usada, o aplicativo estará sujeito à virtualização no Windows Vista, a menos que você
forneça um manifesto do aplicativo em um arquivo de recurso Win32 ou durante uma etapa de build posterior.
No Visual Studio, defina essa opção na página Propriedade do Aplicativo selecionando a opção Criar
aplicativo sem um manifesto na lista suspensa Manifesto. Para obter mais informações, consulte Página
Aplicativo, Designer de Projeto (C#).
Para obter mais informações sobre a criação do manifesto, consulte -win32manifest (Opções do compilador do
C#).
Consulte também
Opções do compilador de C#
Gerenciando propriedades de solução e de projeto
-optimize (opções do compilador C#)
23/10/2019 • 2 minutes to read • Edit Online
A opção -optimize habilita ou desabilita otimizações executadas pelo compilador para tornar o arquivo de saída
menor, mais rápido e mais eficiente.
Sintaxe
-optimize[+ | -]
Comentários
A -optimize também informa o Common Language Runtime para otimizar o código em tempo de execução.
Por padrão, as otimizações estão desabilitadas. Especifique -optimize+ para habilitar otimizações.
Ao criar um módulo a ser usado por um assembly, use as mesmas configurações -optimize que as do assembly.
-o é a forma abreviada de -optimize.
É possível combinar as opções -optimize e -debug.
Para definir esta opção do compilador no ambiente de desenvolvimento do Visual Studio
1. Abra a página Propriedades do projeto.
2. Clique na página de propriedades Compilar.
3. Modifique a propriedade Otimizar código.
Para saber mais sobre como definir essa opção do compilador programaticamente, veja Optimize.
Exemplo
Compile t2.cs e habilite as otimizações do compilador:
Consulte também
Opções do compilador de C#
Gerenciando propriedades de solução e de projeto
-out (opções do compilador C#)
23/10/2019 • 3 minutes to read • Edit Online
Sintaxe
-out:filename
Arguments
filename
O nome do arquivo de saída criado pelo compilador.
Comentários
Na linha de comando, é possível especificar vários arquivos de saída para a compilação. O compilador espera
encontrar um ou mais arquivos de código-fonte após a opção -out. Em seguida, todos os arquivos de código-
fonte serão compilados no arquivo de saída especificado por essa opção -out.
Especifique o nome completo e a extensão do arquivo que você deseja criar.
Se você não especificar o nome do arquivo de saída:
Um .exe extrairá seu nome do arquivo de código-fonte que contém o método principal.
Um arquivo .dll ou .netmodule extrairá seu nome do primeiro arquivo de código-fonte.
Um arquivo de código-fonte usado para compilar um arquivo de saída não pode ser usado na mesma
compilação para a compilação de outro arquivo de saída.
Ao produzir vários arquivos de saída em uma compilação de linha de comando, tenha em mente que somente
um dos arquivos de saída pode ser um assembly e que somente o primeiro arquivo de saída especificado
(implícita ou explicitamente com -out) pode ser o assembly.
Os módulos produzidos como parte de uma compilação se tornam arquivos associados a qualquer assembly
também produzido na compilação. Use ildasm.exe para exibir o manifesto do assembly para ver os arquivos
associados.
A opção do compilador -out é necessária para que um exe seja o destino de um assembly amigável. Para obter
mais informações, consulte Assemblies amigáveis.
Para definir esta opção do compilador no ambiente de desenvolvimento do Visual Studio
1. Abra a página Propriedades do projeto.
2. Clique na página de propriedades do Aplicativo.
3. Modifique a propriedade Nome do Assembly.
Para definir essa opção do compilador de maneira programática: OutputFileName é uma propriedade
somente leitura, determinada por uma combinação do tipo de projeto (exe, biblioteca e assim por diante) e
o nome do assembly. Será necessário modificar uma ou ambas as propriedades para definir o nome do
arquivo de saída.
Exemplo
Compile t.cse crie o arquivo de saída t.exe , bem como compile t2.cs e crie o arquivo de saída de módulo
mymodule.netmodule :
Consulte também
Opções do compilador de C#
Assemblies Amigáveis
Gerenciando propriedades de solução e de projeto
-pathmap (opções do compilador C#)
23/10/2019 • 2 minutes to read • Edit Online
A opção do compilador -pathmap especifica como mapear caminhos físicos para os nomes de caminho de
origem emitidos pelo compilador.
Sintaxe
-pathmap:path1=sourcePath1,path2=sourcePath2
Arguments
path1 O caminho completo para os arquivos de origem no ambiente atual
sourcePath1 O caminho de origem substituído por path1 em arquivos de saída.
Para especificar vários caminhos de origem mapeados, separe-os com uma vírgula.
Comentários
O compilador grava o caminho de origem em sua saída pelos seguintes motivos:
1. O caminho de origem é substituído por um argumento quando o CallerFilePathAttribute é aplicado a um
parâmetro opcional.
2. O caminho de origem é inserido em um arquivo PDB.
3. O caminho do arquivo PDB é inserido em um arquivo PE (executável portátil).
Essa opção mapeia cada caminho físico no computador em que o compilador é executado para um caminho
correspondente que deve ser gravado nos arquivos de saída.
Exemplo
Compilar t.cs no diretório C:\work\tests e mapear esse diretório para \publish na saída:
Consulte também
Opções do compilador de C#
Gerenciando propriedades de solução e de projeto
-pdb (opções do compilador C#)
23/10/2019 • 2 minutes to read • Edit Online
Sintaxe
-pdb:filename
Arguments
filename
O nome e o local do arquivo de símbolos de depuração.
Comentários
Ao especificar -debug (Opções do compilador do C#), o compilador criará um arquivo .pdb no mesmo diretório
em que o compilador criará o arquivo de saída (.exe ou .dll) com um nome de arquivo que é o mesmo que o nome
do arquivo de saída.
O -pdb permite que você especifique um nome de arquivo não padrão e um local para o arquivo .pdb.
Essa opção do compilador não pode ser definida no ambiente de desenvolvimento do Visual Studio nem pode ser
alterada por meio de programação.
Exemplo
Compile t.cs e crie um arquivo .pdb chamado tt.pdb:
Consulte também
Opções do compilador de C#
Gerenciando propriedades de solução e de projeto
-platform (opções do compilador C#)
23/10/2019 • 4 minutes to read • Edit Online
Especifica qual versão do CLR (Common Language Runtime) pode executar o assembly.
Sintaxe
-platform:string
Parâmetros
string
anycpu (padrão), anycpu32bitpreferred, ARM, x64, x86 ou Itanium.
Comentários
O anycpu (padrão) compila seu assembly para que ele seja executado em qualquer plataforma. Seu
aplicativo será executado como um processo de 64 bits sempre que possível e realizará fallback para 32
bits quando apenas esse modo estiver disponível.
anycpu32bitpreferred compila seu assembly para que ele seja executado em qualquer plataforma. Seu
aplicativo é executado no modo 32 bits em sistemas que dão suporte para aplicativos de 32 bits e 64 bits. É
possível especificar essa opção apenas para projetos que definem como destino o .NET Framework 4.5.
O ARM compila seu assembly para que ele seja executado em um computador que tem um processador
ARM (Advanced RISC Machine).
ARM64 compila o assembly para execução pelo CLR de 64 bits em um computador que tem um
processador ARM (Máquina RISC Avançada) que dá suporte ao conjunto de instruções A64.
x64 compila o assembly para ser executado pelo CLR de 64 bits em um computador que dá suporte ao
conjunto de instruções AMD64 ou EM64T.
x86 compila o assembly para ser executado pelo CLR compatível com x86 de 32 bits.
Itanium compila o assembly para ser executado pelo CLR de 64 bits em um computador com um
processador Itanium.
Em um sistema operacional do Windows de 64 bits:
Os assemblies compilados com -platform:x86 são executados no CLR de 32 bits em execução no
WOW64.
Uma DLL compilada com o -platform:anycpu é executada no mesmo CLR que o processo no qual ela é
carregada.
Os executáveis compilados com -platform:anycpu são executados no CLR de 64 bits.
Os executáveis compilados com -platform:anycpu32bitpreferred são executados no CLR de 32 bits.
A configuração anycpu32bitpreferred é válida apenas para arquivos .EXE (executáveis) e requer o .NET
Framework 4.5.
Para obter mais informações sobre o desenvolvimento de um aplicativo para ser executado em um sistema
operacional Windows de 64 bits, consulte Aplicativos de 64 bits.
Para definir esta opção do compilador no ambiente de desenvolvimento do Visual Studio
1. Abra a página Propriedades do projeto.
2. Clique na página de propriedades Compilar.
3. Modifique a propriedade Destino da plataforma e, para projetos que definem como destino o .NET
Framework 4.5, marque ou desmarque a caixa de seleção Preferir 32 bits.
NOTE
-platform Não está disponível no ambiente de desenvolvimento no Visual C# Express.
Para saber mais sobre como definir essa opção do compilador programaticamente, veja PlatformTarget.
Exemplo
O exemplo a seguir mostra como usar a opção -platform para especificar que o aplicativo deve ser executado
pelo CLR de 64 bits em um sistema de operacional Windows de 64 bits.
Consulte também
Opções do compilador de C#
Gerenciando propriedades de solução e de projeto
-preferreduilang (opções do compilador C#)
23/10/2019 • 2 minutes to read • Edit Online
Usando a opção do compilador -preferreduilang , é possível especificar o idioma em que o compilador C# exibe a
saída, como mensagens de erro.
Sintaxe
-preferreduilang: language
Arguments
language
O nome do idioma do idioma a ser usado para a saída do compilador.
Comentários
É possível usar a opção do compilador -preferreduilang para especificar o idioma que você deseja que o
compilador C# use para mensagens de erro e outras saídas de linha de comando. Se o pacote de idiomas do
idioma não estiver instalado, a configuração de idioma do sistema operacional será usada e nenhum erro será
relatado.
csc.exe -preferreduilang:ja-JP
Requisitos
Consulte também
Opções do compilador de C#
-publicsign (opções do compilador C#)
23/10/2019 • 2 minutes to read • Edit Online
Essa opção faz com que o compilador aplique uma chave pública, mas, na verdade, não assina o assembly. A
opção -publicsign também define um bit no assembly que informa o tempo de execução que o arquivo
realmente já está assinado.
Sintaxe
-publicsign
Arguments
nenhuma.
Comentários
A opção -publicsign requer o uso de -keyfile ou -keycontainer. As opções keyfile ou keycontainer especificam
a chave pública.
As opções -publicsign e -delaysign são mutuamente exclusivas.
Às vezes chamada de "assinatura falsa" ou "assinatura de OSS", a assinatura pública inclui a chave pública em um
assembly de saída e define o sinalizador "assinado", mas, na verdade, não assina o assembly com uma chave
privada. Isso é útil para projetos de software livre em que as pessoas desejam criar assemblies compatíveis com
os assemblies "totalmente assinados" lançados, mas não têm acesso à chave privada usada para assinar os
assemblies. Como quase nenhum consumidor precisa verificar realmente se o assembly está totalmente assinado,
esses assemblies criados publicamente podem ser usados em quase todos os cenários nos quais o assembly
totalmente assinado seria usado.
Para definir esta opção do compilador no ambiente de desenvolvimento do Visual Studio
1. Abra a página Propriedades do projeto.
2. Modifique a propriedade Apenas adiar a assinatura.
Consulte também
Opção -delaysign do compilador C#
Opção -keyfile do compilador C#
Opção -keycontainer do compilador C#
Opções do compilador de C#
Gerenciando propriedades de solução e de projeto
-recurse (opções do compilador C#)
23/10/2019 • 2 minutes to read • Edit Online
A opção -recurse permite compilar arquivos de código-fonte em todos os diretórios filho do diretório especificado
(dir) ou do diretório do projeto.
Sintaxe
-recurse:[dir\]file
Arguments
dir (opcional)
O diretório no qual você deseja que a pesquisa comece. Se ele não for especificado, a pesquisa começará no
diretório do projeto.
file
Os arquivos a serem pesquisados. São permitidos caracteres curinga.
Comentários
A opção -recurse permite compilar arquivos de código-fonte em todos os diretórios filho do diretório especificado
( dir ) ou do diretório do projeto.
É possível usar curingas em um nome de arquivo para compilar todos os arquivos correspondentes no diretório
do projeto sem usar -recurse.
Essa opção do compilador não está disponível no Visual Studio e não pode ser alterada programaticamente.
Exemplo
Compila todos os arquivos C# no diretório atual:
csc *.cs
Compila todos os arquivos C# no diretório dir1\dir2 e quaisquer diretórios abaixo dele e gera dir2.dll:
Consulte também
Opções do compilador de C#
Gerenciando propriedades de solução e de projeto
-reference (opções do compilador do C#)
23/10/2019 • 5 minutes to read • Edit Online
A opção -reference opção faz com que o compilador importe informações de tipo public no arquivo
especificado para o projeto atual, permitindo que você referencie metadados dos arquivos do assembly
especificado.
Sintaxe
-reference:[alias=]filename
-reference:filename
Arguments
filename
O nome de um arquivo que contém um manifesto de assembly. Para importar mais de um arquivo, inclua uma
opção -reference separada para cada arquivo.
alias
Um identificador C# válido que representará um namespace raiz que conterá todos os namespaces no
assembly.
Comentários
Para importar de mais de um arquivo, inclua uma opção -reference para cada arquivo.
Os arquivos que você importa devem conter um manifesto; o arquivo de saída deve ter sido compilado com
uma das opções -target diferentes de -target:module.
-r é a forma abreviada de -reference.
Use -addmodule para importar metadados de um arquivo de saída que não contém um manifesto do
assembly.
Se você referenciar um assembly (Assembly A) que referencia outro assembly (Assembly B ), será necessário
referenciar o Assembly B se:
o tipo A que você usar do Assembly A herdar de um tipo ou implementar uma interface do Assembly B.
Você invoca um campo, propriedade, evento ou método que tem um tipo de retorno ou de parâmetro
do Assembly B.
Use -lib para especificar o diretório no qual uma ou mais das suas referências do assembly estão localizadas. O
tópico -lib também aborda os diretórios nos quais o compilador pesquisa assemblies.
Para que o compilador reconheça um tipo em um assembly e não um módulo, ele precisa ser forçado a
resolver o tipo, que você pode fazer definindo uma instância do tipo. Há outras maneiras de resolver nomes de
tipo em um assembly para o compilador: por exemplo, se você herda de um tipo em um assembly, o nome do
tipo será reconhecido pelo compilador.
Às vezes, é necessário referenciar duas versões diferentes do mesmo componente de dentro de um assembly.
Para fazer isso, use a subopção de alias na opção -reference para cada arquivo, a fim de diferenciar entre os
dois arquivos. Esse alias será usado como um qualificador do nome do componente e será resolvido para o
componente em um dos arquivos.
O arquivo de resposta csc (.rsp), que referencia assemblies .NET Framework usados com frequência, é usado
por padrão. Use -noconfig se você não quiser que o compilador use csc.rsp.
NOTE
No Visual Studio, use a caixa de diálogo Adicionar Referência. Para obter mais informações, confira Como: Adicionar ou
remover referências usando o Gerenciador de Referências. Para garantir o comportamento equivalente entre adicionar
referências usando -reference e usando a caixa de diálogo Adicionar Referência, defina a propriedade Inserir Tipos
de Interoperabilidade como False para o assembly que você está adicionando. True é o valor padrão para a
propriedade.
Exemplo
Este exemplo mostra como usar o recurso alias externo.
Compile o arquivo de origem e importe os metadados de grid.dll e de grid20.dll , que foram compilados
anteriormente. As duas DLLs contêm versões separadas do mesmo componente e você usa dois -reference
com opções de alias para compilar o arquivo de origem. As opções têm esta aparência:
-reference:GridV1=grid.dll -reference:GridV2=grid20.dll
Isso configura os aliases externos GridV1 e GridV2 , que você usa em seu programa por meio de uma
instrução extern :
Quando isso for feito, será possível consultar o controle de grade de grid.dll , prefixando o nome do controle
com GridV1 , assim:
GridV1::Grid
Além disso, é possível consultar o controle de grade de grid20.dll , prefixando o nome do controle com
GridV2 , assim:
GridV2::Grid
Consulte também
Opções do compilador de C#
Gerenciando propriedades de solução e de projeto
-refout (Opções do compilador do C#)
24/10/2019 • 2 minutes to read • Edit Online
A opção -refout especifica um caminho de arquivo em que o assembly de referência deve ser gerado. Isso se
converte em metadataPeStream na API de emissão. Essa opção corresponde à propriedade de projeto
ProduceReferenceAssembly do MSBuild.
Sintaxe
-refout:filepath
Arguments
filepath O fipeath so assembly de referência. Geralmente, ele deve corresponder ao assembly principal. A
convenção recomendada (usada pelo MSBuild) é colocar o assembly de referência em uma subpasta "ref/" em
relação ao assembly principal.
Comentários
Os assemblies de referência são um tipo especial de assembly que contém apenas a quantidade mínima de
metadados necessários para representar a superfície da API pública da biblioteca. Eles incluem declarações para
todos os membros que são significativos ao referenciar um assembly em ferramentas de compilação, mas
excluem todas as implementações de membro e declarações de membros privados que não têm impacto
observável em seu contrato de API. Para obter mais informações, consulte Reference Assemblies in .net Guide.
As opções -refout e -refonly são mutualmente exclusivas.
Consulte também
Opções do compilador de C#
Gerenciando propriedades de solução e de projeto
-refonly (Opções do compilador do C#)
24/10/2019 • 2 minutes to read • Edit Online
A opção -refonly indica que um assembly de referência deve ser gerado em vez de um assembly de
implementação, como a saída primária. O parâmetro -refonly silenciosamente desabilita a geração de PDBs,
uma vez que assemblies de referência não podem ser executados. Essa opção corresponde à propriedade de
projeto ProduceOnlyReferenceAssembly do MSBuild.
Sintaxe
-refonly
Comentários
Os assemblies de referência são um tipo especial de assembly que contém apenas a quantidade mínima de
metadados necessários para representar a superfície da API pública da biblioteca. Eles incluem declarações para
todos os membros que são significativos ao referenciar um assembly em ferramentas de compilação, mas
excluem todas as implementações de membro e declarações de membros privados que não têm impacto
observável em seu contrato de API. Para obter mais informações, consulte Reference Assemblies in .net Guide.
As opções -refonly e -refout são mutualmente exclusivas.
Consulte também
Opções do compilador de C#
Gerenciando propriedades de solução e de projeto
-resource (opções do compilador C#)
23/10/2019 • 2 minutes to read • Edit Online
Sintaxe
-resource:filename[,identifier[,accessibility-modifier]]
Arguments
filename
O arquivo de recurso do .NET Framework que você deseja inserir no arquivo de saída.
identifier (opcional)
O nome lógico do recurso; o nome usado para carregar o recurso. O padrão é o nome do nome de arquivo.
accessibility-modifier (opcional)
A acessibilidade do recurso: público ou privado. O padrão é público.
Comentários
Use -linkresource para vincular um recurso a um assembly e não adicionar o arquivo de recurso ao arquivo de
saída.
Por padrão, recursos são públicos no assembly quando são criados usando o compilador C#. Para tornar os
recursos privados, especifique private como o modificador de acessibilidade. Não é permitida nenhuma outra
acessibilidade diferente de public ou private .
Se filename for um arquivo de recurso do .NET Framework criado, por exemplo, por Resgen.exe ou no
ambiente de desenvolvimento, ele poderá ser acessado com membros no namespace System.Resources. Para
obter mais informações, consulte System.Resources.ResourceManager. Para todos os outros recursos, use os
métodos GetManifestResource na classe Assembly para acessar o recurso em tempo de execução.
-res é a forma abreviada de -resource.
A ordem dos recursos no arquivo de saída é determinada com base na ordem especificada na linha de comando.
Para definir esta opção do compilador no ambiente de desenvolvimento do Visual Studio
1. Adicionar um arquivo de recurso ao seu projeto.
2. Selecione o arquivo que você deseja inserir no Gerenciador de Soluções.
3. Selecione Ação de Build para o arquivo na janela Propriedades.
4. Defina Ação de Build como Recurso inserido.
Para saber mais sobre como definir essa opção do compilador programaticamente, veja BuildAction.
Exemplo
Compile in.cs e anexe ao arquivo de recurso rf.resource :
csc -resource:rf.resource in.cs
Consulte também
Opções do compilador de C#
Gerenciando propriedades de solução e de projeto
-subsystemversion (opções do compilador C#)
23/10/2019 • 3 minutes to read • Edit Online
Especifica a versão mínima do subsistema no qual o arquivo executável gerado pode ser executado, determinando
assim as versões do Windows em que o arquivo executável pode ser executado. Normalmente, essa opção
garante que o arquivo executável possa tirar proveito de determinados recursos de segurança que não estão
disponíveis com versões mais antigas do Windows.
NOTE
Para especificar o subsistema em si, use a opção do compilador -target.
Sintaxe
-subsystemversion:major.minor
Parâmetros
major.minor
A versão mínima obrigatória do subsistema, conforme expresso em uma notação de ponto para versões principais
e secundárias. Por exemplo, você pode especificar que um aplicativo não pode ser executado em um sistema
operacional mais antigo que o Windows 7 se definir o valor dessa opção como 6.01, conforme descrito na tabela
mais adiante neste tópico. Você deve especificar os valores de major e minor como inteiros.
Zeros à esquerda na versão minor não alteram a versão, mas zeros à direita alteram. Por exemplo, 6.1 e 6.01 se
referem à mesma versão, mas 6.10 se refere a uma versão diferente. É recomendável expressar a versão
secundária como dois dígitos para evitar confusão.
Comentários
A seguinte tabela lista as versões de subsistema comuns do Windows.
Windows XP 5.01
Windows 7 6.01
Windows 8 6.02
Valores padrão
O valor padrão da opção do compilador -subsystemversion depende das condições na lista a seguir:
O valor padrão é 6.02 se qualquer opção do compilador na lista a seguir for definida:
/target:appcontainerexe
/target:winmdobj
-platform:arm
O valor padrão será 6.00 se você estiver usando o MSBuild, se tiver como destino o .NET Framework 4.5 e
se não definiu nenhuma das opções de compilador que foram especificadas anteriormente na lista.
O valor padrão é 4.00 se nenhuma das condições anteriores for verdadeira.
Consulte também
Opções do compilador de C#
-target (opções do compilador C#)
27/11/2019 • 2 minutes to read • Edit Online
A opção do compilador -target pode ser especificada em uma das quatro formas:
/target:appcontainerexe
Para criar um arquivo. exe para aplicativos da loja do Windows 8. x.
/target:exe
Para criar um arquivo .exe.
/target:library
Para criar uma biblioteca de códigos.
/target:module
Para criar um módulo.
/target:winexe
Para criar um programa do Windows.
/target:winmdobj
Para criar um arquivo .winmdobj intermediário.
A menos que você especifique -target:module, o -target faz com que um manifesto do assembly do .NET
Framework seja colocado em um arquivo de saída. Para obter mais informações, consulte assemblies no .net e
atributos comuns.
O manifesto do assembly é colocado no primeiro arquivo de saída .exe na compilação ou na primeira DLL, se
não houver nenhum arquivo de saída .exe. Por exemplo, na linha de comando a seguir, o manifesto será
colocado em 1.exe :
O compilador cria apenas um manifesto do assembly por compilação. As informações sobre todos os arquivos
em uma compilação são colocados no manifesto do assembly. Todos os arquivos de saída, exceto os criados
com -target:module podem conter um manifesto do assembly. Ao gerar vários arquivos de saída na linha de
comando, apenas um manifesto do assembly pode ser criado e ele deve ir para o primeiro arquivo de saída
especificado na linha de comando. Não importa qual é o primeiro arquivo de saída ( -target:exe, -
target:winexe, -target:library ou -target:module), todos os outros arquivos de saída produzidos na mesma
compilação devem ser módulos ( -target:module).
Se você criar um assembly, pode indicar que todo ou parte do seu código está em conformidade com CLS com
o atributo CLSCompliantAttribute.
// target_clscompliant.cs
[assembly:System.CLSCompliant(true)] // specify assembly compliance
Consulte também
Opções do compilador de C#
Gerenciando propriedades de solução e de projeto
-subsystemversion (opções do compilador do C#)
-target:appcontainerexe (opções do compilador C#)
27/11/2019 • 2 minutes to read • Edit Online
Se você usar a opção do compilador -target:appcontainerexe, o compilador criará um arquivo executável (.exe)
do Windows que deverá ser executado em um contêiner de aplicativos. Essa opção é equivalente a -target: winexe
, mas é projetada para aplicativos da loja do Windows 8. x.
Sintaxe
-target:appcontainerexe
Comentários
Para exigir que o aplicativo seja executado em um contêiner de aplicativos, esta opção define um bit no arquivo
PE. Quando esse bit estiver definido, ocorrerá um erro se o método CreateProcess tentar inicializar o arquivo
executável fora de um contêiner de aplicativos.
A menos que você use a opção -out, o nome do arquivo de saída usará o nome do arquivo de entrada que
contém o método Main.
Quando você especifica essa opção em um prompt de comando, todos os arquivos até a próxima opção -out ou -
target são usados para criar o arquivo executável.
Para definir esta opção do compilador no IDE
1. No Gerenciador de Soluções, abra o menu de atalho do projeto e escolha Propriedades.
2. Na guia Aplicativo, na lista Tipo de saída, escolha Aplicativo da Windows Store.
Essa opção está disponível somente para modelos de aplicativo da loja do Windows 8. x.
Para saber mais sobre como definir essa opção do compilador programaticamente, veja OutputType.
Exemplo
O comando a seguir compila filename.cs em um arquivo executável do Windows que pode ser executado
somente em um contêiner de aplicativos.
Consulte também
-target (opções do compilador do C#)
-target:winexe (opções do compilador do C#)
Opções do compilador de C#
-target:exe (opções do compilador C#)
23/10/2019 • 2 minutes to read • Edit Online
A opção -target:exe faz com que o compilador crie um aplicativo de console executável (EXE ).
Sintaxe
-target:exe
Comentários
A opção -target:exe está em vigor por padrão. O arquivo executável será criado com a extensão .exe.
Use -target:winexe para criar um executável de programa do Windows.
A menos que seja especificado de outra forma com a opção -out, o nome do arquivo de saída usará o nome do
arquivo de entrada que contém o método Main.
Quando especificado na linha de comando, todos os arquivos até a próxima opção -out ou -target:module
serão usados para criar o arquivo .exe
Somente um método Main é necessário nos arquivos de código-fonte que são compilados em um arquivo .exe.
A opção do compilador -main permite especificar qual classe contém o método Main, nos casos em que o código
tem mais de uma classe com um método Main.
Para definir esta opção do compilador no ambiente de desenvolvimento do Visual Studio
1. Abra a página Propriedades do projeto.
2. Clique na página de propriedades do Aplicativo.
3. Modifique a propriedade Tipo de saída.
Para saber mais sobre como definir essa opção do compilador programaticamente, veja OutputType.
Exemplo
Cada uma das seguintes linhas de comando compilará in.cs , criando in.exe :
Consulte também
-target (opções do compilador do C#)
Opções do compilador de C#
-target:library (opções do compilador C#)
23/10/2019 • 2 minutes to read • Edit Online
A opção -target:library faz com que o compilador crie uma DLL (biblioteca de vínculo dinâmico) em vez de um
EXE (arquivo executável).
Sintaxe
-target:library
Comentários
A DLL será criada com a extensão .dll.
A menos que seja especificado de outra forma com a opção -out, o nome do arquivo de saída usa o nome do
primeiro arquivo de entrada.
Quando especificado na linha de comando, todos os arquivos até a próxima opção -out ou -target:module são
usados para criar o arquivo .dll.
Ao criar um arquivo .dll, um método Main não é necessário.
Para definir esta opção do compilador no ambiente de desenvolvimento do Visual Studio
1. Abra a página Propriedades do projeto.
2. Clique na página de propriedades do Aplicativo.
3. Modifique a propriedade Tipo de saída.
Para saber mais sobre como definir essa opção do compilador programaticamente, veja OutputType.
Exemplo
Compile in.cs , criando in.dll :
Consulte também
-target (opções do compilador do C#)
Opções do compilador de C#
-target:module (opções do compilador C#)
23/10/2019 • 2 minutes to read • Edit Online
Essa opção faz com que o compilador não gere um manifesto do assembly.
Sintaxe
-target:module
Comentários
Por padrão, o arquivo de saída criado por meio da compilação com essa opção terá uma extensão .netmodule.
Um arquivo que não tem um manifesto do assembly não pode ser carregado pelo Common Language
Runtime do .NET Framework. No entanto, esse arquivo pode ser incorporado no manifesto do assembly de
um assembly por meio de -addmodule.
Se mais de um módulo for criado em uma única compilação, tipos internos em um módulo estarão disponíveis
para outros módulos na compilação. Quando o código em um módulo referenciar tipos internal em outro
módulo, os dois módulos deverão ser incorporados em um manifesto do assembly, por meio de -addmodule.
Não há suporte para a criação de um módulo no ambiente de desenvolvimento do Visual Studio.
Para saber mais sobre como definir essa opção do compilador programaticamente, veja OutputType.
Exemplo
Compile in.cs , criando in.netmodule :
Consulte também
-target (opções do compilador do C#)
Opções do compilador de C#
-target:winexe (opções do compilador C#)
23/10/2019 • 2 minutes to read • Edit Online
A opção -target:winexe faz com que o compilador crie um programa do Windows executável (EXE ).
Sintaxe
-target:winexe
Comentários
O arquivo executável será criado com a extensão .exe. Um programa do Windows é aquele que fornece uma
interface do usuário da biblioteca do .NET Framework ou com as APIs do Windows.
Use -target:exe para criar um aplicativo do console.
A menos que seja especificado de outra forma com a opção -out, o nome do arquivo de saída usará o nome do
arquivo de entrada que contém o método Main.
Quando especificado na linha de comando, todos os arquivos até a próxima opção -out ou -target serão usados
para criar o programa do Windows.
Somente um método Main é necessário nos arquivos de código-fonte que são compilados em um arquivo .exe.
A opção -main permite especificar qual classe contém o método Main, nos casos em que o código tem mais de
uma classe com um método Main.
Para definir esta opção do compilador no ambiente de desenvolvimento do Visual Studio
1. Abra a página Propriedades do projeto.
2. Clique na página de propriedades do Aplicativo.
3. Modifique a propriedade Tipo de saída.
Para saber mais sobre como definir essa opção do compilador programaticamente, veja OutputType.
Exemplo
Compile in.cs em um programa do Windows:
Consulte também
-target (opções do compilador do C#)
Opções do compilador de C#
-target:winmdobj (opções do compilador C#)
27/11/2019 • 3 minutes to read • Edit Online
Sintaxe
-target:winmdobj
Comentários
A configuração winmdobj indica para o compilador que um módulo intermediário é necessário. Em resposta, o
Visual Studio compila a biblioteca de classes do C# como um arquivo .winmdobj. O arquivo .winmdobj pode,
então, ser alimentado por meio da ferramenta de exportação WinMDExp para produzir um arquivo de
metadados do Windows (.winmd). O arquivo .winmd contém o código da biblioteca original e os metadados do
WinMD que são usados pelo JavaScript ou C++ e pelo Windows Runtime.
A saída de um arquivo que é compilado usando a opção do compilador -target:winmdobj foi criada para ser
usada apenas como entrada para a ferramenta de exportação WimMDExp. O arquivo .winmdobj em si não é
referenciado diretamente.
A menos que você use a opção -out, o nome do arquivo de saída usará o nome do primeiro arquivo de entrada.
Um método Main não é necessário.
Se você especificar a opção -target:winmdobj em um prompt de comando, todos os arquivos até a próxima opção
-out ou -target:module serão usados para criar o programa do Windows.
Para definir esta opção de compilador no IDE do Visual Studio para um aplicativo da Windows Store
1. No Gerenciador de Soluções, abra o menu de atalho do projeto e escolha Propriedades.
2. Escolha a guia Aplicativo.
3. Na lista Tipo de saída, escolha Arquivo WinMD.
A opção de arquivo WinMD está disponível somente para modelos de aplicativo da loja do Windows 8. x.
Para saber mais sobre como definir essa opção do compilador programaticamente, veja OutputType.
Exemplo
O comando a seguir compila filename.cs em um arquivo .winmdobj intermediário.
Consulte também
-target (opções do compilador do C#)
Opções do compilador de C#
-unsafe (opções do compilador C#)
23/10/2019 • 2 minutes to read • Edit Online
A opção do compilador -unsafe permite que o código que usa a palavra-chave unsafe seja compilado.
Sintaxe
-unsafe
Comentários
Para obter mais informações sobre código não seguro, consulte Código não seguro e ponteiros.
Para definir esta opção do compilador no ambiente de desenvolvimento do Visual Studio
1. Abra a página Propriedades do projeto.
2. Clique na página de propriedades Compilar.
3. Marque a caixa de seleção Permitir código não seguro.
Para adicionar essa opção em um arquivo csproj
Abra o arquivo .csproj de um projeto e adicione os seguintes elementos:
<PropertyGroup>
<AllowUnsafeBlocks>true</AllowUnsafeBlocks>
</PropertyGroup>
Para saber mais sobre como definir essa opção do compilador programaticamente, veja AllowUnsafeBlocks.
Exemplo
Compile in.cs para o modo não seguro:
Consulte também
Opções do compilador de C#
Gerenciando propriedades de solução e de projeto
-utf8output (opções do compilador C#)
23/10/2019 • 2 minutes to read • Edit Online
Sintaxe
-utf8output
Comentários
Em algumas configurações internacionais, a saída do compilador não pode ser exibida corretamente no console.
Nessas configurações, use -utf8output e redirecione a saída do compilador para um arquivo.
Essa opção do compilador não está disponível no Visual Studio e não pode ser alterada programaticamente.
Consulte também
Opções do compilador de C#
-warn (opções do compilador C#)
23/10/2019 • 2 minutes to read • Edit Online
Sintaxe
-warn:option
Arguments
option
O nível de aviso que você deseja exibir para a compilação: Números mais baixos mostram apenas avisos de
gravidade alta; números mais altos mostram mais avisos. Os valores válidos vão de 0 a 4:
Comentários
Para obter informações sobre um erro ou aviso, você pode procurar o código de erro no Índice da Ajuda. Para
outras maneiras de se obter informações sobre um erro ou aviso, consulte Erros do compilador do C#.
Use -warnaserror para tratar todos os avisos como erros. Use -nowarn para desabilitar determinados avisos.
-w é a forma abreviada de -warn.
Para definir esta opção do compilador no ambiente de desenvolvimento do Visual Studio
1. Abra a página Propriedades do projeto.
2. Clique na página de propriedades Compilar.
3. Modifique a propriedade Nível de Aviso.
Para saber mais sobre como definir essa opção do compilador programaticamente, veja WarningLevel.
Exemplo
Compilar in.cs e fazer com que o compilador exiba somente avisos de nível 1:
Consulte também
Opções do compilador de C#
Gerenciando propriedades de solução e de projeto
-warnaserror (opções do compilador C#)
23/10/2019 • 2 minutes to read • Edit Online
Sintaxe
-warnaserror[+ | -][:warning-list]
Comentários
Quaisquer mensagens que seriam, normalmente, ser relatadas como avisos, são relatadas como erros e o
processo de build é interrompido (os arquivos de saída não são compilados).
Por padrão, a opção -warnaserror- está em vigor, o que faz com que os avisos não impeçam a geração de um
arquivo de saída. A -warnaserror, que é a mesma que -warnaserror+ , faz com que os avisos sejam tratados
como erros.
Opcionalmente, se você deseja que apenas avisos específicos sejam tratados como erros, você pode especificar
uma lista separada por vírgulas de números de aviso para serem tratados como erros.
Use -warn para especificar o nível de avisos que você deseja que o compilador exiba. Use -nowarn para desabilitar
determinados avisos.
Para definir esta opção do compilador no ambiente de desenvolvimento do Visual Studio
1. Abra a página Propriedades do projeto.
2. Clique na página de propriedades Compilar.
3. Modifique a propriedade Tratar Avisos como Erros.
Para definir programaticamente essa opção do compilador, confira TreatWarningsAsErrors.
Exemplo
Compilar in.cs e fazer com que o compilador não exiba avisos:
Consulte também
Opções do compilador de C#
Gerenciando propriedades de solução e de projeto
-win32icon (opções do compilador C#)
23/10/2019 • 2 minutes to read • Edit Online
A opção -win32icon insere um arquivo .ico no arquivo de saída, que fornece ao arquivo de saída a aparência
desejada no Explorador de Arquivos.
Sintaxe
-win32icon:filename
Arguments
filename
O arquivo .ico que você deseja adicionar ao seu arquivo de saída.
Comentários
Um arquivo .ico pode ser criado com o Compilador de Recurso. O Compilador de Recurso é invocado quando
você compila um programa do Visual C++; um arquivo .ico é criado com base no arquivo .rc.
Consulte -linkresource (para referenciar) ou -resource (para anexar) um arquivo de recurso do .NET Framework.
Consulte -win32res para importar um arquivo .res.
Para definir esta opção do compilador no ambiente de desenvolvimento do Visual Studio
1. Abra a página de Propriedades do projeto.
2. Clique na página de propriedades do Aplicativo.
3. Modifique a propriedade Ícone do aplicativo.
Para saber mais sobre como definir essa opção do compilador programaticamente, veja ApplicationIcon.
Exemplo
Compilar in.cs e anexar um arquivo .ico rf.ico para produzir in.exe :
Consulte também
Opções do compilador de C#
Gerenciando propriedades de solução e de projeto
-win32manifest (opções do compilador C#)
23/10/2019 • 4 minutes to read • Edit Online
Use a opção -win32manifest para especificar um arquivo de manifesto do aplicativo Win32 definido pelo
usuário para ser inserido em um arquivo PE do projeto.
Sintaxe
-win32manifest: filename
Arguments
filename
O nome e o local do arquivo de manifesto personalizado.
Comentários
Por padrão, o compilador do Visual C# insere um manifesto do aplicativo que especifica o nível de execução
solicitado "asInvoker". Ele cria o manifesto na mesma pasta em que o executável é compilado, normalmente a
pasta bin\Debug ou bin\Release quando você usa o Visual Studio. Se você quiser fornecer um manifesto
personalizado, por exemplo, para especificar um nível de execução solicitado de "highestAvailable" ou
"requireAdministrator", use esta opção para especificar o nome do arquivo.
NOTE
Essa opção e a opção -win32res (opções do compilador do C#) são mutuamente exclusivas. Se você tentar usar ambas as
opções na mesma linha de comando, você obterá um erro de build.
Um aplicativo que não tem nenhum manifesto do aplicativo que especifica um nível de execução solicitado estará
sujeito à virtualização de arquivos/Registro sob o recurso Controle de Conta de Usuário no Windows. Para obter
mais informações, consulte Controle de Conta de Usuário.
Seu aplicativo estará sujeito à virtualização se alguma dessas condições for verdadeira:
Você usa a opção -nowin32manifest e não fornece um manifesto em uma etapa de build posterior ou
como parte de um arquivo de Recurso (.res) do Windows usando a opção -win32res.
Você fornece um manifesto personalizado que não especifica um nível de execução solicitado.
O Visual Studio cria um arquivo .manifest padrão e o armazena nos diretórios de depuração e liberação
juntamente com o arquivo executável. Você pode adicionar um manifesto personalizado criando um em qualquer
editor de texto e, em seguida, adicionando o arquivo ao projeto. Como alternativa, você pode clicar com o botão
direito do mouse no ícone Projeto no Gerenciador de Soluções, clicar em Adicionar Novo Item e clicar em
Arquivo de Manifesto do Aplicativo. Depois de adicionar o arquivo de manifesto novo ou existente, ele
aparecerá na lista suspensa Manifesto. Para obter mais informações, consulte Página Aplicativo, Designer de
Projeto (C#).
Você pode fornecer o manifesto do aplicativo como uma etapa de pós-build personalizada ou como parte de um
arquivo de recurso Win32 usando a opção -nowin32manifest (opções do compilador do C#). Use essa mesma
opção se quiser que o aplicativo seja sujeito à virtualização de arquivo ou Registro no Windows Vista. Isso
impedirá que o compilador crie e insira um manifesto padrão no arquivo PE.
Exemplo
O exemplo a seguir mostra o manifesto padrão que o Compilador do Visual C# insere em um PE.
NOTE
O compilador insere um nome de aplicativo padrão "MyApplication" no xml. Essa é uma solução alternativa para habilitar os
aplicativos para serem executados no Windows Server 2003 Service Pack 3.
Consulte também
Opções do compilador de C#
-nowin32manifest (opções do compilador do C#)
Gerenciando propriedades de solução e de projeto
-win32res (opções do compilador C#)
23/10/2019 • 2 minutes to read • Edit Online
Sintaxe
-win32res:filename
Arguments
filename
O arquivo de recurso que você deseja adicionar ao seu arquivo de saída.
Comentários
Um arquivo de recurso do Win32 pode ser criado com o Compilador de Recursos. O Compilador de Recursos é
invocado quando você compila um programa do Visual C++; um arquivo .res é criado com base no arquivo .rc.
Um recurso do Win32 pode conter informações de versão ou de bitmap (ícone) que ajudariam a identificar seu
aplicativo no Explorador de Arquivos. Se você não especificar a -win32res, o compilador gerará informações de
versão com base na versão do assembly.
Consulte -linkresource (para referenciar) ou -resource (para anexar) um arquivo de recurso do .NET Framework.
Para definir esta opção do compilador no ambiente de desenvolvimento do Visual Studio
1. Abra a página Propriedades do projeto.
2. Clique na página de propriedades do Aplicativo.
3. Clique no botão Arquivo de Recurso e escolha um arquivo usando a caixa de combinação.
Exemplo
Compilar in.cs e anexar um arquivo de recurso do Win32 rf.res para produzir in.exe :
Consulte também
Opções do compilador de C#
Gerenciando propriedades de solução e de projeto
Erros do compilador de C#
24/10/2019 • 2 minutes to read • Edit Online
Alguns erros do compilador do C# têm tópicos correspondentes que explicam por que o erro é gerado e, em
alguns casos, como corrigir o erro. Use uma das etapas a seguir para ver se a ajuda está disponível para uma
mensagem de erro específica.
Localize o número do erro (por exemplo, CS0029) na Janela de Saída e pesquise por ele no Microsoft
Docs.
Escolha o número do erro (por exemplo, CS0029) na Janela de Saída e, em seguida, pressione a tecla F1.
No Índice, digite o número do erro na caixa Pesquisar.
Se nenhuma dessas etapas levar a informações sobre o erro, vá para o fim desta página e envie comentários que
incluam o número ou o texto do erro.
Para obter informações sobre como configurar opções de aviso e erro em C#, consulte Compilar página, Designer
de Projeto (C#).
NOTE
Seu computador pode mostrar diferentes nomes ou locais para alguns dos elementos de interface do usuário do Visual
Studio nas instruções a seguir. A edição do Visual Studio que você possui e as configurações que você usa determinam esses
elementos. Para obter mais informações, consulte Personalizando o IDE.
Consulte também
Opções do compilador de C#
Infelizmente, não há informações específicas sobre este erro de C#
Página de Build, Designer de Projeto (C#)
-warn (opções do compilador do C#)
-nowarn (opções do compilador do C#)
Introdução
101 minutes to read • Edit Online
Hello world
O programa "Hello, World" é usado tradicionalmente para introduzir uma
linguagem de programação. Este é para C#:
using System;
class Hello
{
static void Main() {
Console.WriteLine("Hello, World");
}
}
csc hello.cs
Hello, World
O programa "Hello, World" começa com uma diretiva using que faz referência
ao namespace System . Namespaces fornecem um meio hierárquico de
organizar bibliotecas e programas em C#. Os namespaces contêm tipos e
outros namespaces — por exemplo, o namespace System contém uma
quantidade de tipos, como a classe Console referenciada no programa e
diversos outros namespaces, como IO e Collections . A diretiva using que
faz referência a um determinado namespace permite o uso não qualificado dos
tipos que são membros desse namespace. Devido à diretiva using , o
programa pode usar Console.WriteLine como um atalho para
System.Console.WriteLine .
Estrutura do programa
Os principais conceitos organizacionais em C# são programas, namespaces,
tipos, membros e assemblies. Os programas C# consistem em um ou mais
arquivos de origem. Os programas declaram tipos que contêm membros e
podem ser organizados em namespaces. Classes e interfaces são exemplos de
tipos. Campos, métodos, propriedades e eventos são exemplos de membros.
Quando os programas em C# são compilados, eles são empacotados
fisicamente em assemblies. Normalmente, os assemblies têm a .exe extensão
.dll de arquivo ou, dependendo se eles implementam aplicativos ou
bibliotecas.
O exemplo
using System;
namespace Acme.Collections
{
public class Stack
{
Entry top;
class Entry
{
public Entry next;
public object data;
using System;
using Acme.Collections;
class Test
{
static void Main() {
Stack s = new Stack();
s.Push(1);
s.Push(10);
s.Push(100);
Console.WriteLine(s.Pop());
Console.WriteLine(s.Pop());
Console.WriteLine(s.Pop());
}
}
100
10
1
Tipos e variáveis
Há dois tipos em C#: tipos de referência e tipos de valor. As variáveis de tipos
de valor contêm diretamente seus dados enquanto variáveis de tipos de
referência armazenam referências a seus dados, o último sendo conhecido
como objetos. Com tipos de referência, é possível que duas variáveis
referenciem o mesmo objeto e, portanto, é possível que operações em uma
variável afetem o objeto referenciado por outra variável. Com tipos de valor,
cada variável tem sua própria cópia dos dados e não é possível que operações
em uma variável afetem a outra (exceto no caso de variáveis de parâmetros
ref e out ).
CATEGORIA DESCRIÇÃO
Caracteres Unicode:
char
Booliano: bool
Cadeia de caracteres
Unicode: string
Os tipos integrais oito dão suporte a valores de 8 bits, 16 bits, 32 bits e 64 bits
no formulário com ou sem sinal.
Os dois tipos de ponto flutuante float , double e, são representados usando
os formatos IEEE 754 de precisão única de 32 bits e de 64 bits de precisão
dupla.
O tipo decimal é um tipo de dados de 128 bits adequado para cálculos
financeiros e monetários.
C# false o bool tipoéusadopararepresentarvaloresBoolianos—
valoresquesãoou. true
O processamento de cadeia de caracteres e caracteres em C# usa codificação
Unicode. O tipo char representa uma unidade de código UTF -16 e o tipo
string representa uma sequência de unidades de código UTF -16.
INTERVALO/PRECIS
CATEGORIA BITS TIPO ÃO
32 int -2147483648... 2,
147, 483, 647
INTERVALO/PRECIS
CATEGORIA BITS TIPO ÃO
64 long -
9.223.372.036.854
.775.808... 9, 223,
372, 036, 854,
775, 807
Os tipos anuláveis também não precisam ser declarados antes que possam ser
usados. Para cada tipo T de valor não anulável, há um tipo T? anulável
correspondente, que pode conter um valor null adicional. Por exemplo, int?
é um tipo que pode conter qualquer número inteiro de 32 bits ou null o valor.
C#o sistema de tipos do é unificado, de modo que um valor de qualquer tipo
possa ser tratado como um objeto. Cada tipo no C#, direta ou indiretamente,
deriva do tipo de classe object , e object é a classe base definitiva de todos os
tipos. Os valores de tipos de referência são tratados como objetos
simplesmente exibindo os valores como tipo object . Os valores dos tipos de
valor são tratados como objetos executando operações Boxing e unboxing .
No exemplo a seguir, um valor int é convertido em object e volta
novamente ao int .
using System;
class Test
{
static void Main() {
int i = 123;
object o = i; // Boxing
int j = (int)o; // Unboxing
}
}
Expressões
Expressões são construídas a partir de operandos e operadores. Os
operadores de uma expressão indicam quais operações devem ser aplicadas
aos operandos. Exemplos de operadores incluem + , - , * , / e new .
Exemplos de operandos incluem literais, campos, variáveis locais e expressões.
Quando uma expressão contiver vários operadores, a precedência dos
operadores controla a ordem na qual os operadores individuais são avaliados.
Por exemplo, a expressão x + y * z é avaliada como x + (y * z) porque o
operador * tem precedência maior do que o operador + .
A maioria dos operadores pode ser sobrecarregada. A sobrecarga de operador
permite que implementações de operador definidas pelo usuário sejam
especificadas para operações em que um ou ambos os operandos são de um
tipo struct ou de classe definida pelo usuário.
A tabela a seguir resume C#os operadores, listando as categorias de operador
em ordem de precedência, da mais alta para a mais baixa. Operadores na
mesma categoria têm a mesma precedência.
x++ Pós-incremento
x-- Pós-decremento
Unário +x Identidade
-x Negação
!x Negação lógica
++x Pré-incremento
CATEGORIA EXPRESSÃO DESCRIÇÃO
--x Pré-decremento
Multiplicativo x * y Multiplicação
x / y Divisão
x % y Restante
x - y Subtração, remoção de
delegado
Igualdade x == y Igual a
x != y Diferente de
Instruções
As ações de um programa são expressas usando instruções. O C# oferece
suporte a vários tipos diferentes de instruções, algumas delas definidas em
termos de instruções inseridas.
Um bloco permite a produção de várias instruções em contextos nos quais
uma única instrução é permitida. Um bloco é composto por uma lista de
instruções escritas entre os delimitadores { e } .
Instruções de declaração são usadas para declarar constantes e variáveis
locais.
Instruções de expressão são usadas para avaliar expressões. As expressões
que podem ser usadas como instruções incluem invocações de método,
alocações de objeto usando o new operador, atribuições = usando e os
operadores de atribuição compostos, operações de incremento e decréscimo
usando o ++ operadores -- e e expressões Await.
Instruções de seleção são usadas para selecionar uma dentre várias instruções
possíveis para execução com base no valor de alguma expressão. Neste grupo
estão as instruções if e switch .
Instruções de iteração são usadas para executar repetidamente uma instrução
inserida. Neste grupo estão as instruções while , do , for e foreach .
Instruções de salto são usadas para transferir o controle. Neste grupo estão as
instruções break , continue , goto , throw , return e yield .
A instrução try ... catch é usada para capturar exceções que ocorrem durante
a execução de um bloco, e a instrução try ... finally é usada para especificar
o código de finalização que é executado sempre, se uma exceção ocorrer ou
não.
As checked instruções unchecked e são usadas para controlar o contexto de
verificação de estouro para operações aritméticas de tipo integral e conversões.
A instrução lock é usada para obter o bloqueio de exclusão mútua para um
determinado objeto, executar uma instrução e, em seguida, liberar o bloqueio.
A instrução using é usada para obter um recurso, executar uma instrução e,
em seguida, descartar esse recurso.
Veja abaixo exemplos de cada tipo de instrução
Declarações de variáveis locais
Instrução de expressão
Instrução if
Instrução switch
static void Main(string[] args) {
int n = args.Length;
switch (n) {
case 0:
Console.WriteLine("No arguments");
break;
case 1:
Console.WriteLine("One argument");
break;
default:
Console.WriteLine("{0} arguments", n);
break;
}
}
Instrução while
Instrução do
Instrução for
Instrução foreach
Instrução break
static void Main() {
while (true) {
string s = Console.ReadLine();
if (s == null) break;
Console.WriteLine(s);
}
}
Instrução continue
Instrução goto
Instrução return
Instrução yield
Instrução lock
class Account
{
decimal balance;
public void Withdraw(decimal amount) {
lock (this) {
if (amount > balance) {
throw new Exception("Insufficient funds");
}
balance -= amount;
}
}
}
Instrução using
Classes e objetos
As classes são os tipos do C# mais fundamentais. Uma classe é uma estrutura
de dados que combina ações (métodos e outros membros da função) e estado
(campos) em uma única unidade. Uma classe fornece uma definição para
instâncias da classe criadas dinamicamente, também conhecidas como
objetos. As classes dão suporte à herança e polimorfismo, mecanismos nos
quais classes derivadas podem estender e especializar classes base.
Novas classes são criadas usando declarações de classe. Uma declaração de
classe inicia com um cabeçalho que especifica os atributos e modificadores de
classe, o nome da classe, a classe base (se fornecida) e as interfaces
implementadas pela classe. O cabeçalho é seguido pelo corpo da classe, que
consiste em uma lista de declarações de membro escrita entre os delimitadores
{ e } .
Instâncias de classes são criadas usando o operador new , que aloca memória
para uma nova instância, chama um construtor para inicializar a instância e
retorna uma referência à instância. As instruções a seguir criam Point dois
objetos e armazenam referências a esses objetos em duas variáveis:
ASSOCIADO DESCRIÇÃO
Acessibilidade
Cada membro de uma classe tem uma acessibilidade associada, que controla
as regiões de texto do programa que são capazes de acessar o membro. Há
cinco formas possíveis de acessibilidade. Esses métodos estão resumidos na
tabela a seguir.
ACESSIBILIDADE SIGNIFICADO
Parâmetros de tipo
Uma definição de classe pode especificar um conjunto de parâmetros de tipo
seguindo o nome da classe com colchetes angulares com uma lista de nomes
de parâmetro de tipo. Os parâmetros de tipo podem ser usados no corpo das
declarações de classe para definir os membros da classe. No exemplo a seguir,
os parâmetros de tipo de Pair são TFirst e TSecond :
public class Pair<TFirst,TSecond>
{
public TFirst First;
public TSecond Second;
}
Uma classe herda os membros de sua classe base. Herança significa que uma
classe implicitamente contém todos os membros de sua classe base, exceto
para os construtores estáticos e de instância, e os destruidores da classe base.
Uma classe derivada pode adicionar novos membros aos que ela herda, mas
ela não pode remover a definição de um membro herdado. No exemplo
anterior, Point3D herda os campos x e y de Point e cada instância
Point3D contém três campos: x , y e z .
Campos
Um campo é uma variável associada a uma classe ou com uma instância de
uma classe.
Um campo declarado com o static modificador define um campo estático.
Um campo estático identifica exatamente um local de armazenamento. Não
importa quantas instâncias de uma classe são criadas, há sempre apenas uma
cópia de um campo estático.
Um campo declarado sem o static modificador define um campo de
instância. Cada instância de uma classe contém uma cópia separada de todos
os campos de instância dessa classe.
No exemplo a seguir, cada instância da classe Color tem uma cópia separada
dos campos de instância r , g e b , mas há apenas uma cópia dos campos
estáticos Black , White , Red , Green e Blue :
using System;
class Test
{
static void Swap(ref int x, ref int y) {
int temp = x;
x = y;
y = temp;
}
class Test
{
static void Divide(int x, int y, out int result, out int remainder) {
result = x / y;
remainder = x % y;
}
using System;
class Squares
{
static void Main() {
int i = 0;
int j;
while (i < 10) {
j = i * i;
Console.WriteLine("{0} x {0} = {1}", i, j);
i = i + 1;
}
}
}
public Entity() {
serialNo = nextSerialNo++;
}
using System;
class Test
{
static void Main() {
Entity.SetNextSerialNo(1000);
Entity e1 = new Entity();
Entity e2 = new Entity();
Console.WriteLine(e1.GetSerialNo()); // Outputs "1000"
Console.WriteLine(e2.GetSerialNo()); // Outputs "1001"
Console.WriteLine(Entity.GetNextSerialNo()); // Outputs "1002"
}
}
using System;
using System.Collections;
class Test
{
static void Main() {
Expression e = new Operation(
new VariableReference("x"),
'*',
new Operation(
new VariableReference("y"),
'+',
new Constant(2)
)
);
Hashtable vars = new Hashtable();
vars["x"] = 3;
vars["y"] = 5;
Console.WriteLine(e.Evaluate(vars)); // Outputs "21"
vars["x"] = 1.5;
vars["y"] = 9;
Console.WriteLine(e.Evaluate(vars)); // Outputs "16.5"
}
}
Sobrecarga de método
A sobrecarga de método permite que vários métodos na mesma classe tenham
o mesmo nome, contanto que tenham assinaturas exclusivas. Ao compilar uma
invocação de um método sobrecarregado, o compilador usa a resolução de
sobrecarga para determinar o método específico para invocar. A resolução de
sobrecarga localizará o método que melhor corresponde aos argumentos ou
relatará um erro se nenhuma correspondência for encontrada. O exemplo a
seguir mostra a resolução de sobrecarga em vigor. O comentário para cada
invocação no método Main mostra qual método é realmente chamado.
class Test
{
static void F() {
Console.WriteLine("F()");
}
// Fields...
T[] items;
int count;
// Constructors...
public List(int capacity = defaultCapacity) {
public List(int capacity = defaultCapacity) {
items = new T[capacity];
}
// Properties...
public int Count {
get { return count; }
}
public int Capacity {
get {
return items.Length;
}
set {
if (value < count) value = count;
if (value != items.Length) {
T[] newItems = new T[value];
Array.Copy(items, 0, newItems, 0, count);
items = newItems;
}
}
}
// Indexer...
public T this[int index] {
get {
return items[index];
}
set {
items[index] = value;
OnChanged();
}
}
// Methods...
public void Add(T item) {
if (count == Capacity) Capacity = count * 2;
items[count] = item;
count++;
OnChanged();
}
protected virtual void OnChanged() {
if (Changed != null) Changed(this, EventArgs.Empty);
}
public override bool Equals(object other) {
return Equals(this, other as List<T>);
}
static bool Equals(List<T> a, List<T> b) {
if (a == null) return b == null;
if (b == null || a.count != b.count) return false;
for (int i = 0; i < a.count; i++) {
if (!object.Equals(a.items[i], b.items[i])) {
return false;
}
}
return true;
}
// Event...
public event EventHandler Changed;
// Operators...
public static bool operator ==(List<T> a, List<T> b) {
return Equals(a, b);
}
public static bool operator !=(List<T> a, List<T> b) {
return !Equals(a, b);
}
}
Construtores
O C# dá suporte aos construtores estáticos e de instância. Um construtor de
instância é um membro que implementa as ações necessárias para inicializar
uma instância de uma classe. Um construtor estático é um membro que
implementa as ações necessárias para inicializar uma classe quando ele for
carregado pela primeira vez.
Um construtor é declarado como um método sem nenhum tipo de retorno e o
mesmo nome que a classe continente. Se uma declaração de Construtor incluir
static um modificador, ele declara um construtor estático. Caso contrário, ela
declara um construtor de instância.
Construtores de instância podem ser sobrecarregados. Por exemplo, a classe
List<T> declara dois construtores de instância, um sem parâmetros e um que
utiliza um parâmetro int . Os construtores de instância são invocados usando
o operador new . As instruções a seguir alocam duas List<string> instâncias
usando cada um dos construtores List da classe.
Indexadores
Um indexador é um membro que permite que objetos sejam indexados da
mesma forma que uma matriz. Um indexador é declarado como uma
propriedade, exceto se o nome do membro for this seguido por uma lista de
parâmetros escrita entre os delimitadores [ e ] . Os parâmetros estão
disponíveis nos acessadores do indexador. Semelhante às propriedades, os
indexadores podem ser de leitura-gravação, somente leitura e somente
gravação, e os acessadores de um indexador pode ser virtuais.
A classe List declara um indexador único de leitura-gravação que usa um
parâmetro int . O indexador possibilita indexar instâncias List com valores
int . Por exemplo
using System;
class Test
{
static int changeCount;
class Test
{
static void Main() {
List<int> a = new List<int>();
a.Add(1);
a.Add(2);
List<int> b = new List<int>();
b.Add(1);
b.Add(2);
Console.WriteLine(a == b); // Outputs "True"
b.Add(3);
Console.WriteLine(a == b); // Outputs "False"
}
}
Structs
Como classes, os structs são estruturas de dados que podem conter membros
de dados e os membros da função, mas, ao contrário das classes, as estruturas
são tipos de valor e não precisam de alocação de heap. Uma variável de um
tipo struct armazena diretamente os dados do struct, enquanto que uma
variável de um tipo de classe armazena uma referência a um objeto alocado
dinamicamente. Os tipos de estrutura não dão suporte à herança especificada
pelo usuário, e todos os tipos de structs são herdados implicitamente do tipo
object .
class Point
{
public int x, y;
class Test
{
static void Main() {
Point[] points = new Point[100];
for (int i = 0; i < 100; i++) points[i] = new Point(i, i);
}
}
struct Point
{
public int x, y;
Construtores de struct são invocados com o operador new , mas isso não
significa que a memória está sendo alocada. Em vez de alocar dinamicamente
um objeto e retornar uma referência a ele, um construtor de struct
simplesmente retorna o valor do struct (normalmente em um local temporário
na pilha), e esse valor é, então, copiado conforme necessário.
Com classes, é possível que duas variáveis referenciem o mesmo objeto e,
portanto, é possível que operações em uma variável afetem o objeto
referenciado por outra variável. Com structs, as variáveis têm sua própria cópia
dos dados e não é possível que as operações em um afetem o outro. Por
exemplo, a saída produzida pelo fragmento de código a seguir depende de
Point se ser uma classe ou uma estrutura.
Matrizes
Uma matriz é uma estrutura de dados que contém algumas variáveis
acessadas por meio de índices calculados. As variáveis contidas em uma matriz,
também chamadas de elementos da matriz, são todas do mesmo tipo, e esse
tipo é chamado de tipo de elemento da matriz.
Os tipos de matriz são tipos de referência, e a declaração de uma variável de
matriz simplesmente reserva espaço para uma referência a uma instância de
matriz. As instâncias de matriz reais são criadas dinamicamente em tempo de
execução new usando o operador. A new operação especifica o comprimento
da nova instância de matriz, que é então corrigida para o tempo de vida da
instância. Os índices dos elementos de uma matriz variam de 0 a Length - 1 .
O operador new inicializa automaticamente os elementos de uma matriz
usando o valor padrão, que, por exemplo, é zero para todos os tipos numéricos
e null para todos os tipos de referência.
O exemplo a seguir cria uma matriz de elementos int , inicializa a matriz e
imprime o conteúdo da matriz.
using System;
class Test
{
static void Main() {
int[] a = new int[10];
for (int i = 0; i < a.Length; i++) {
a[i] = i * i;
}
for (int i = 0; i < a.Length; i++) {
Console.WriteLine("a[{0}] = {1}", i, a[i]);
}
}
}
A primeira linha cria uma matriz com três elementos, cada um do tipo int[] , e
cada um com um valor inicial de null . As linhas subsequentes inicializam os
três elementos com referências às instâncias individuais da matriz de tamanhos
variados.
O new operador permite que os valores iniciais dos elementos da matriz sejam
especificados usando um inicializador de matriz, que é uma lista de
expressões gravadas entre { os delimitadores e. } O exemplo a seguir aloca
e inicializa um int[] com três elementos.
Interfaces
Uma interface define um contrato que pode ser implementado por classes e
estruturas. Uma interface pode conter métodos, propriedades, eventos e
indexadores. Uma interface não fornece implementações dos membros que
define — ela simplesmente especifica os membros que devem ser fornecidos
por classes ou estruturas que implementam a interface.
As interfaces podem empregar a herança múltipla. No exemplo a seguir, a
interface IComboBox herda de ITextBox e IListBox .
interface IControl
{
void Paint();
}
interface IDataBound
{
void Bind(Binder b);
}
Em casos nos quais uma instância não é conhecida por ser estática para
implementar uma interface específica, é possível usar conversões de tipo
dinâmico. Por exemplo, as instruções a seguir usam conversões dinâmicas de
tipo para obter IControl as IDataBound implementações de um objeto e de
interface. Como o tipo real do objeto é, EditBox as conversões são realizadas
com sucesso.
Enums
Um tipo enum é um tipo de valor diferente com um conjunto de constantes
nomeadas. O exemplo a seguir declara e usa um tipo enum chamado Color
com três valores constantes, Red , Green e Blue .
using System;
enum Color
{
Red,
Green,
Blue
}
class Test
{
static void PrintColor(Color color) {
switch (color) {
case Color.Red:
Console.WriteLine("Red");
break;
case Color.Green:
Console.WriteLine("Green");
break;
case Color.Blue:
Console.WriteLine("Blue");
break;
default:
Console.WriteLine("Unknown color");
break;
}
}
Color c = 0;
Delegados
Um delegado é um tipo que representa referências aos métodos com uma lista
de parâmetros e tipo de retorno específicos. Delegados possibilitam o
tratamento de métodos como entidades que podem ser atribuídos a variáveis e
passadas como parâmetros. Os delegados são parecidos com o conceito de
ponteiros de função em outras linguagens, mas ao contrário dos ponteiros de
função, os delegados são orientados a objetos e fortemente tipados.
O exemplo a seguir declara e usa um tipo delegado chamado Function .
using System;
class Multiplier
{
double factor;
class Test
{
static double Square(double x) {
return x * x;
}
using System;
[Help("http://msdn.microsoft.com/.../MyClass.htm")]
public class Widget
{
[Help("http://msdn.microsoft.com/.../MyClass.htm", Topic = "Display")]
public void Display(string text) {}
}
using System;
using System.Reflection;
class Test
{
static void ShowHelp(MemberInfo member) {
HelpAttribute a = Attribute.GetCustomAttribute(member,
typeof(HelpAttribute)) as HelpAttribute;
if (a == null) {
Console.WriteLine("No help for {0}", member);
}
else {
Console.WriteLine("Help for {0}:", member);
Console.WriteLine(" Url={0}, Topic={1}", a.Url, a.Topic);
}
}
O passo a passo fornece instruções detalhadas para cenários comuns, o que os torna um bom local para começar a
aprender sobre o produto ou uma área de recurso específica.
Esta seção contém links para o passo a passo de programação C#.
Nesta seção
Acessando a Web usando Async e Await
Mostra como criar uma solução assíncrona usando async e await.
Criando um componente do Windows Runtime em C# ou no Visual Basic e chamando-o do JavaScript
Mostra como criar um tipo de Windows Runtime, empacotá-lo em um componente Windows Runtime e, em
seguida, chamar o componente de um aplicativo da loja do Windows 8. x criado para o Windows usando
JavaScript.
Programação do Office (C# e Visual Basic)
Mostra como criar uma planilha do Excel e um documento do Word usando C# e Visual Basic.
Criando e usando objetos dinâmicos (C# e Visual Basic)
Mostra como criar um objeto personalizado que expõe dinamicamente o conteúdo de um arquivo de texto e como
criar um projeto que usa a biblioteca IronPython .
Criando um controle composto com o Visual C#
Demonstra como criar um controle composto simples e estender sua funcionalidade por meio da herança.
Criando um controle do Windows Forms que aproveita os recursos de tempo de design do Visual Studio
Ilustra como criar um designer personalizado para um controle personalizado.
Herdando um controle do Windows Forms com Visual C#
Demonstra como criar um controle de botão herdado simples. Esse botão herda a funcionalidade do botão padrão
do Windows Forms e expõe um membro personalizado.
Depurando controles personalizados do Windows Forms em tempo de design
Descreve como depurar o comportamento de tempo de design de seu controle personalizado.
Executando tarefas comuns usando smart tags em controles do Windows Forms
Demonstra algumas das tarefas mais comuns, como a adição ou remoção de uma guia em um TabControl ,o
encaixe de um controle em seu pai e a alteração da orientação de um controle SplitContainer .
Escrevendo consultas em C# (LINQ )
Demonstra os recursos de linguagem C# que são usados para gravar expressões de consulta LINQ.
Manipulando dados (C#) (LINQ to SQL )
Descreve um cenário LINQ to SQL para adicionar, modificar e excluir dados em um banco de dados.
Modelo de objeto simples e consulta (C#) (LINQ to SQL )
Demonstra como criar uma classe de entidade e uma consulta simples para filtrar a classe de entidade.
Usando somente procedimentos armazenados (C#) (LINQ to SQL )
Demonstra como usar LINQ to SQL para acessar dados executando somente os procedimentos armazenados.
Consultando através de relações (C#) (LINQ to SQL )
Demonstra o uso de associações LINQ to SQL para representar as relações de chave estrangeira em um banco de
dados.
Escrevendo um visualizador em C#
Mostra como escrever um visualizador simples usando C#.
Seções Relacionadas
Exemplos e instruções passo a passo para implantações
Fornece exemplos passo a passo de cenários comuns de implantação.
Consulte também
Guia de Programação em C#
Exemplos do Visual Studio