
Bon Nada! – Feliz Navidad – Merry Christmas 2011

Sin la intención de profundizar excesivamente y relativo a una pregunta relativa al control de la ejecución de varias instancias de una misma aplicación. Se me ocurre responder : Con el uso MUTEX!
Creo que vale la pena que ‘echéis’ un vistazo al código… reamente infalible!
bool SingleInstance; Mutex mutex = new Mutex(false, "MiCodigoMutex", out SingleInstance); if (!SingleInstance) { MessageBox.Show(null, "No se puede ejecutar mas de una instancia",
"Programa en ejecucion", MessageBoxButtons.OK, MessageBoxIcon.Stop); this.Close(); }
Saludos navideños,
PepLluis,
Atendiendo el comentario de Jose Fco en el que indica :
<< He podido leer diversos ejemplos de como utilizar e implementar las funciones básicas de acceso al puerto serie principalmente en aplicaciones WinForms, seria posible obtener una esqueleto base en WPF tal como tienes para los anteriores.>>
El siguiente ejemplo, se entiende como ejercicio para realizar la transmisión y recepción de una trama utilizando un temporizador de un segundo en un proyecto WPF, para cerrar el circuito deberás insertar un conector en el puerto serie con un puente entre los pines 2 y 3. Recordad que el uso de ReadLine suspende la ejecucion del thread hasta recibir la sequencia definida en ‘SerialPort.NewLine’.
Partial Public Class Window1
Delegate Sub Actualizar() Private WithEvents Puerto1 As New System.IO.Ports.SerialPort Private temporizador As New System.Timers.Timer Private Recibidos As String = ""
Sub New() ' Llamada necesaria para el Diseñador de Windows Forms. InitializeComponent() Puerto1 = My.Computer.Ports.OpenSerialPort("COM1", 9600) AddHandler Puerto1.DataReceived, AddressOf Recepcion temporizador.Interval = 1000 temporizador.Enabled = True AddHandler temporizador.Elapsed, AddressOf Peticion End Sub
Private Sub Recepcion() Recibidos = Puerto1.ReadLine Me.Dispatcher.Invoke(New Actualizar(AddressOf Datos)) End Sub
Sub Peticion() Puerto1.WriteLine("Saludos!") End Sub
Sub Datos() Me.Label1.Content = Recibidos End Sub End Class
Espero que cumpla con tus espectativas.
Saludos,
PepLluis,
Es complicado poder atender todas las solicitudes recibidas en este blog, a menudo intento responder a vuestras inquietudes a pesar de que tambien siempre intento convenceros de que utilizemos los Foros de MSDN puesto que es el sitio adecuado para poder compartir vuestras preguntas para que todos los expertos puedan aportar su opinion.
Para preguntas sobre el puerto serie os recomiendo visitar : http://social.msdn.microsoft.com/Forums/es-ES/devcommes/threads
Una vez aclarado 🙂
Bolivar pregunta : a luis tenia una consulta , como puedo convertir cualquier numero entero introducido en un textbox para despues convertirlo a hexadecimal y luego enviarlo por RS232
Repuesta:
Private MiPuertoSerie As New IO.Ports.SerialPort Sub EnviarHex(Numero As String) Numero = Hex(Val(Numero)).ToString MiPuertoSerie.Write(Numero) End Sub
Respondiendo una pregunta de los foros, me complace compartir con vosotros el siguiente ejercicio; no sin antes puntualizar que se trata de la puesta en escena de un ejemplo que utilizaría un modulo como “clase estática” donde depositar objetos compartidos dentro de nuestro espacio de aplicación.
Para evitar interpretaciones, este es uno de esos ejemplos que no debe considerarse “modelo” pues su escenario es poco menos que atípico, aclarar que en situaciones similares lo adecuado seria diseñar una clase completa con toda la funcionalidad y acceso, para posteriormente incluirla como propiedad en un modulo. La dificultad se encuentra en la cantidad de código que deberíamos desplegar para realizar una explicación que finalmente terminaría siendo compleja de seguir y entender.
Dicho esto, el siguiente es solo un punto de entrada para las personas que intentan iniciarse en el tema y se encuentran en situaciones como la que expone “Caudiz” en los foros… necesita poder acceder al puerto serie desde dos forms diferentes.
Primero definiríamos el modulo que contendrá toda la lógica de acceso al puerto serie y podría ser algo similar a lo que sigue :
Imports System.IO.Ports Module SerialComm Private SerialPort1 As New SerialPort Private LastFrameRead As String Public Event DataAvailable() Sub New() SerialPort1.PortName = "COMx" SerialPort1.Open() AddHandler SerialPort1.DataReceived, AddressOf ReceiveData End Sub Private Sub ReceiveData() LastFrameRead += SerialPort1.ReadExisting.ToString() RaiseEvent DataAvailable() End Sub Public Function ReadData() As String Return LastFrameRead End Function Public Sub WriteData(Frame) SerialPort1.Write(Frame) End Sub Public Sub ClearData() LastFrameRead = "" End Sub End Module
Finalmente incluiríamos el siguiente código para acceder a dicha funcionalidad en este caso desde el ‘Form1’ aunque evidentemente podemos incluirlo desde cualquier otro punto del espacio de nuestra aplicación.
Public Class Form1 Sub Form1_Load(sender As System.Object, e As System.EventArgs) Handles MyBase.Load AddHandler SerialComm.DataAvailable, AddressOf ReadData Form2.Show() End Sub Sub ReadData() Me.Label1.Text = SerialComm.ReadData End Sub Private Sub Button1_Click(sender As System.Object, e As System.EventArgs) Handles Button1.Click SerialComm.WriteData("Hola :-) from 1") End Sub End Class
Public Class Form2 Private Sub Form2_Load(sender As System.Object, e As System.EventArgs) Handles MyBase.Load AddHandler SerialComm.DataAvailable, AddressOf ReadData End Sub Sub ReadData() Me.Label1.Text = SerialComm.ReadData End Sub Private Sub Button1_Click(sender As System.Object, e As System.EventArgs) Handles Button1.Click SerialComm.WriteData("Hola :-) from 2") End Sub End Class
Espero vuestros comentarios y que a partir de esta idea, podáis construir de forma robusta la implementación de vuestra solucion.
Saludos,
PepLluis,
Supongo que como “Tips and Tricks” por pregunta recurrente, cuando intentamos implementar un Servicio “WCF” en nuestro servidor, después de haberlo probado repetidamente en nuestro equipo de desarrollo sin ningún tipo de incidencia…
A tu PREGUNTA (Pablo):
Cuando intento conectarme al servicio desplegado en mi servidor, después de haber pasado las pruebas en nuestro entorno de desarrollo, recibo el siguiente mensaje : "El servicio no autenticó al autor de la llamada”, pero en mi caso no me interesa ningún tipo de autenticación para poder acceder al servicio, como debo configurar el cliente?.
La RESPUESTA:
A banda de que en este caso concreto no te interese implementar ningún tipo de autenticación es casi obligado que leas las recomendaciones sobre la arquitectura de seguridad para WCF en .NET Framework 4.0, este es el link : http://msdn.microsoft.com/es-es/library/ms788756.aspx
En tu caso el mensaje te esta indicando que tu servicio web esta implementando un “wsHttpBinding”, Si lo que te interesa es un acceso “Básico” para entendernos como con los antiguos “ws”, entonces debes editar el archivo de configuración del host “web.config” donde tengas instalado el servicio, y modificar del <System.ServiceModel> el <endpoint address="" binding="wsHttpBinding" contract="TusServicios.IService1"> por :
binding="basicHttpBinding" … con ‘b’ minúscula 🙂
Luego es muy importante que actualices las referencias a tu servicio, pues en caso contrario el cliente se responderá con una excepción :
“No se pudo activar el servicio solicitado”… es obvio por la discordancia, desde el explorador de soluciones, click en el servicio y seleccionas la opción
– Actualizar referencia de Servicio
y Listo.
Saludos,
PepLluis,
Es curioso cómo puede variar el significado de las palabras… al oír este nombre en código, no puedo evitar visualizar imágenes de mi juventud, en concreto cuando los hombres del campo se acercaban al bar para robarle un respiro al día y apoyándose en la barra pedían a voces “un café” “una Copa” y un “Rosly’n” gesticulando transcendentalmente… jajaja, No sé si por casualidad o coincidencia todos los machotes que practicaron esa poco saludable costumbre, terminaron como los compiladores… con las cajas torácicas bien negras.
Dicho esto, parece ser que el término se invierte y ahora “Roslyn” representara el fin de tres décadas de cajas negras invocando los prompt’s de un misterioso mago escondido tras un ejecutable llamado compilador.
Este es otro de esos excitantes puntos de inflación en los lenguajes, a pesar de los “roadmap” de producto, evoluciones y tecnologías que Visual Studio va incorporando a la plataforma, sin duda este paso me recuerda a un nuevo hito que cambiara nuestro concepto a la par que lo hizo Linq en su momento. Por lo tanto “Bienvenido Roslyn”
¿Pero qué es Roslyn?
Es la posibilidad de utilizar los compiladores de C# y Visual Basic invocándolos en un modelo de servicios. Lo se una respuesta simple, difícil de asimilar.
Durante muchos años cientos de programadores han estado reclamando la posibilidad de disponer de “compiladores dinámicos” a los que poder invocar sin necesidad de llamar complejos ‘scripts’ dentro o fuera del propio entorno de programación.
Anders Hejlsberg, dio una charla sobre el futuro C# y Visual Basic en el pasado BUILD, os dejo el enlace por si acaso: http://channel9.msdn.com/Events/BUILD/BUILD2011/TOOL-816T
Os aconsejo descargaros la CTP de octubre 2011, es divertidísimo trastear con ella, además tendrás acceso a un montón de ‘Walkthrough’ entre los cuales encontraras el proyecto para generarte la extensión que te permitirá jugar a ser Anders, copiando código entre dos proyectos distintos; uno en VB y pegando en C#, poco menos que genial!!
Para ello solo tenéis que compilar el proyecto “ConvertingPasteExtensions” incluido en el CTP y luego instalar la extensión generada en Visual Studio… una vez completado, podrás disponer del genial “Paste as C#” o “Paste as Visual Basic”.
Aunque es este sentido, la implementación no está realizada al completo, deberéis tener paciencia pues no todas las expresiones “sobre todo Linq” funcionaran de un lado a otro… pero como digo, eso no quita las ganas de empezar a inventar descubriendo el montón de nuevas posibilidades que “Roslyn” nos podrá al descubierto.
Puedes descargarte la CTP desde: http://msdn.microsoft.com/es-es/roslyn
Que lo disfrutéis!,
PepLluis,
Atendiendo la petición de un lector pidiéndome el código completo…
Public Class Form1 ''' <summary> ''' Complete code to test the frame reading values sample (last post) ''' PepLluis, 11/10/2011 ''' </summary> ''' <param name="sender"></param> ''' <param name="e"></param> ''' <remarks></remarks> Private Sub Form1_Load(sender As System.Object, e As System.EventArgs) Handles MyBase.Load ' Simulate string frame with values Dim frame = "{W111}{M222}{F333}{S444}" ' define Label and textboxes Dim Label1 As New Label With {.Name = "Label1", .AutoSize = True, .Location = New Drawing.Point(12, 10), .Text = "Frame :" + frame} Dim TextBox1 As New TextBox With {.Name = "TextBox1", .Width = 60, .Location = New Drawing.Point(12, 30)} Dim TextBox2 As New TextBox With {.Name = "TextBox2", .Width = 60, .Location = New Drawing.Point(12, 50)} Dim TextBox3 As New TextBox With {.Name = "TextBox3", .Width = 60, .Location = New Drawing.Point(12, 70)} Dim TextBox4 As New TextBox With {.Name = "TextBox4", .Width = 60, .Location = New Drawing.Point(12, 90)} ' add controls to form Me.Controls.AddRange({Label1, TextBox1, TextBox2, TextBox3, TextBox4}) ' define char as fields separators and values iden Dim separators As Char() = {"{", "W", "M", "S", "F", "}"} ' get fields Dim fields As String() = ( From f As String In frame.Split(separators) Where f.Length > 0 ).ToArray ' get textBoxes names Dim textBoxesNames() = ( From Tbn As Object In Me.Controls Where Tbn.GetType.Name = "TextBox" Select CType(Tbn, TextBox).Name Order By Name ).ToArray ' assign values to textboxes For Index As Integer = 0 To textBoxesNames.Count - 1 Me.Controls.Item(textBoxesNames(Index)).Text = fields(Index) Next End Sub End Class
Regards! :-)
PepLluis,
Perdido en la recepción de datos. A menudo recibo vuestras preguntas pidiéndome si os puedo aportar alguna idea para resolver casuísticas concretas. Lamentablemente por falta de tiempo no siempre puedo dedicar un post a cada una de vuestras inquietudes, sin embargo en ocasiones se plantean situaciones que se repiten y en caso de resolverlas considero que pueden ayudar a más personas con la misma inquietud y creo que este es el caso : Pedro me escribe con lo siguiente: <Estoy encallado en un punto del cual me está costando salir… tengo un hardware especifico por el que el puerto serie me envía una trama conteniendo la información de varios valores separados por unos delimitadores> La pregunta es: Como puedo detectar cuando recibo los indicadores, como distingo la letra que identifica el dato y finalmente como puedo leer únicamente su valor y colocarlo en un textbox. |
I’m lost receiving RS232 data. I often get questions asking me if I can contribute with any idea to solve concrete cases. Unfortunately due to lack of time I cannot always devote a post to each of your concerns, however sometimes there are situations that are repeated in case of solving them believe that they can help more people with the same concern and provably that may be the case: Peter write me with the next question : < I am aground at a point which is costing me leave; I have an specific hardware with a serial port, that port sends me a frame containing information from several values separated by a few delimiters>
The question is: How I can detect the indicators positions, and how I can distinguish the letter that identifies the data and finally how can read only its value and send this to a textbox.
|
Visual basic =
Public Class Form1 Private Sub Form1_Load(sender As System.Object, e As System.EventArgs) Handles MyBase.Load Dim frame = "{W111}{M22}{F33}{S44}" Dim separators As Char() = {"{", "W", "M", "S", "F", "}"} Dim fields As String() = (From f As String In frame.Split(separators) Where f.Length > 0).ToArray Dim textBoxesNames() = (From Tbn As Object In Me.Controls Where Tbn.GetType.Name = "TextBox" Select CType(Tbn, TextBox).Name Order By Name).ToArray For Index As Integer = 0 To textBoxesNames.Count - 1 Me.Controls.Item(textBoxesNames(Index)).Text = fields(Index) Next End Sub End Class
C# =
using System; using System.Linq; using System.Windows.Forms; namespace WindowsFormsApplication1 { public partial class Form1 : Form { public Form1() { InitializeComponent(); } private void Form1_Load(object sender, EventArgs e) { string frame = "{W111}{M22}{F33}{S44}"; char[] separators = new char[] { '{', 'W', 'M', 'S', 'F', '}' }; string[] fields = (from string f in frame.Split(separators) where f.Length > 0 select f).ToArray(); string[] textBoxesNames = (from Tbn in this.Controls.OfType<TextBox>() where Tbn.GetType().Name == "TextBox" orderby Tbn.Name select Tbn.Name).ToArray(); for (int Index = 0; Index < textBoxesNames.Count(); Index++) { this.Controls[textBoxesNames[Index]].Text = fields[Index]; } } } }
Regards,
PepLluis,
A propósito de mi último post, mi encuentro con el equipo de tres desarrolladores propicio una discusión en torno a cómo controlar el número de versión de cada proyecto, después de discutir llano y largo lo oportuno de realizar estas tareas de una manera centralizada desde los gestores de “Control de Código” y dando por imposible entenderlo de otra forma, se me planteo el reto de cómo poder incrementar el número de versión del ensamblado del proyecto desde el propio código del proyecto de forma automática y en función a si se estaba haciendo un ‘debug’ o un ‘release’.
Evidentemente el siguiente código, muestra y demuestra que es posible aunque debo rehuir de cualquier otra interpretación para tales prácticas, pues creo que es evidente que salvo excepciones no pueden ser tomadas más que como un ejercicio “curioso”.
using System; using System.Text.RegularExpressions; using System.Windows.Forms; namespace VersionString { public partial class Form1 : Form { public Form1() { InitializeComponent(); } private void Form1_Load(object sender, EventArgs e) { SetNewVersion(); LblVersion.Text = Application.ProductVersion.ToString(); } private void SetNewVersion() { try { string AssemblyFile = @"..\..\Properties\AssemblyInfo.cs"; string str = System.IO.File.ReadAllText(AssemblyFile); Regex r = new Regex(@"\[assembly\: AssemblyVersion\(""(\d{1,})\.(\d{1,})\.(\d{1,})\.(\d{1,})""\)\]"); Match m = r.Match(str); string rz = string.Format("[assembly: {4}(\"{0}.{1}.{2}.{3}\")]" , m.Groups[1].Value , m.Groups[2].Value , m.Groups[3].Value , Convert.ToString((Convert.ToInt16(m.Groups[4].Value))+1) , "AssemblyVersion"); System.IO.File.WriteAllText(AssemblyFile, r.Replace(str, rz)); } catch (Exception ex) { MessageBox.Show(ex.Message); } } } }
A falta de convencer al equipo para que se animen a utilizar algunos de los excelentes gestores de código para visual studio, aquí va mi dedicatoria… a demás con el código reclamo esa ronda de cervezas. J
PepLluis,