mercoledì 13 aprile 2011

C++ e C# a confronto

Molto spesso si è portati a pensare che C#, il linguaggio "di punta" del framework .NET, sia un C++ "migliore", un "C++++" come suggerisce l'immagine, in realtà le differenze sono profonde, più che da un punto di vista sintattico, da un punto di vista concettuale. 
Vediamo di seguito una tabella sulle maggiori differenze riguardo i due linguaggi.

C++
C#
I riferimenti ad un’altra classe in un altro file del progetto richiedono un #include
Nessun #include è richiesto
I riferimenti ad una classe in una libreria dinamica (dll) richiedono di includere un file e il link ad una libreria statica (lib)
Non ci sono più le librerie statiche. Si possono aggiungere i riferimenti alle dll (assemblies) direttamente da progetto
I simboli esportati da una libreria devono essere dichiarati come declspec o in un .DEF file
Basta dichiarare i simboli come pubblici nella libreria. I simboli privati non vengono esportati.
Se ci si riferisce ad una classe/struttura prima che venga dichiarata, bisogna fare una forward declaration
Non esistono le forward declarations e l’ordine di dichiarazione non è importante
Passando tra stringhe Unicode/16-bit si richiede:
  • entry point differenti (main/wmain)
  • Prefissi non standard “L” or “_T”, oppure gestione STL con string/wstring
  • Le stringhe vengono manipolate da differenti funzioni
Unicode è il formato nativo
NULL è un concetto, non fa parte del linguaggio. È tipicamente definito come 0 e quindi è l'equivalente di 0.
null è una parola chiave. Non è equivalente a 0.
I tipi di dato base (int, float, char, ecc) non possono accettare un valore "NULL". Solo i puntatori possono avere un valore "NULL".
Tutti i tipi C # possono accettare null come  valore
• T, con il supporto null è “T?”, ad es. int con il supporto null è “int?”
Nessun supporto per “foreach”
Supporto di foreach per iterare su una collezione
Supporto per try e catch.
Nessun supporto per "finally”
Supporto per try-catch-finally

Nessuna notifica su operazioni di overflow.
Per esempio:
unsigned x = UINT_MAX;
x++;
// x == 0 nessuna exception lanciata
Usando le parole chiave checked/unchecked puoi ottenere comportamenti differenti (exception/overflow)
Nessun supporto nativo per sezioni critiche o multithreading
Parola chiave lock per le critical sections, oggetti Thread built-in
All’interno di uno switch, il break è opzionale
Il break dentro lo switch è obbligatorio
Nessuna documentazione built-in. Esite la possibilità di usare tools esterni come doxygen
Documentazione XML con “///”. Riconosciuto da IntelliSense in Visual Studio
Non si può suddividere la definizione di una classe su più file
Le classi parziali permettono di suddividere la definizione di una classe su più source files.

Nessun supporto per mix di librerie di versioni differenti
Si può fare un mix di librerie di versioni differenti

Lo sviluppatore deve gestire la memoria direttamente (new/delete)

Supporto di Garbage collection
Si ha maggiore controllo sulla distruzione degli oggetti, possibilità di memory leaks se non ben gestita
Minore controllo sulla distruzione degli oggetti, ma più sicurezza a causa della gestione automatica del GC
Gli oggetti “reference” possono essere allocati sia nell’heap che nello stack della memoria dichiarandoli come locali o con una new
Gli oggetti “reference” vengono sempre allocati nell’heap
Supporto per classi o metodi “friend”
Non esiste il concetto di friend, ma questo è stato in qualche modo sostituito dalle Proprietà di una classe
I tipi non condividono un tipo root comune

Tutti i tipi derivano dalla classe “Object”


Pensandoci bene e leggendo vari articoli sull’argomento, ci si rende conto che in realtà la differenza tra un linguaggio come C++ e uno come C# (o Java), è ben più profonda e parte dalla base stessa di come è stato creato il linguaggio.
Il C++ si potrebbe definire come un linguaggio a due livelli di astrazione:

  • Livello 1 (low level): gestione diretta della memoria e dell’allocazione delle risorse attraverso i puntatori . A questo livello viene definita la classe come detentrice delle risorse che verranno rilasciate nel suo distruttore. L’acquisizione-inizializzazione-distruzione delle risorse usate dalla classe è completamente a carico del programmatore che deve essere quindi sufficientemente esperto per progetti di medie-grosse dimensioni.
  • Livello 2 (high level): a questo livello si vedono solo gli “oggetti” e si possono usare le risorse solo attraverso i metodi esposti dagli “oggetti”. Non esistono puntatori e nemmeno “delete” o “free”: al limite esistono puntatori “intelligenti” detti smart_pointer che si auto gestiscono e distruggono.  Si richiede un grado di esperienza del programmatore anche inferiore perché la gestione “delicata” delle risorse è fatta al livello sottostante. Questo livello è molto simile a quello offerto da C#/Java.

Le origini del C++ non partono esclusivamente dal C, di cui comunque si è scelto di mantenere la compatibilità, ma soprattutto da un’idea di Stroustrup che il linguaggio stesso potesse essere esteso, partendo da un nucleo decisamente ridotto all'osso. Il programmatore può decidere di costruire sul linguaggio nativo estensioni difficilmente distinguibili dal linguaggio stesso. Un chiaro esempio di ciò è la libreria STL.
Il C#, come il Java, lasciano meno possibilità e libertà al programmatore, che probabilmente però ne guadagna in “sicurezza”e velocità di apprendimento. C# come Java partono da un livello di astrazione superiore al C++ e però se da un lato ciò comporta l’evitare tutte quelle problematiche che possono insorgere quando si programma a basso livello, dall’altro ne perde in flessibilità e potenza espressiva. C’è da dire che probabilmente la scelta di Java/C# è oculata perchè il target dei linguaggi è molto differente: nel caso di Java/C# si hanno linguaggi destinati ad applicazioni che devono interagire pesantemente col mondo della rete, del web, dei database, delle applicazioni distribuite ecc., per cui si è scelto di semplificare la vita del programmatore dando un livello di astrazione maggiore: in questo modo è stato possibile costruire e far apprendere in tempi relativamente rapidi ambienti quali .NET o Java (inteso come ambiente). Il C++ è stato pensato invece per andare a coprire target quali applicazioni time critical o sistemi embedded: target per cui è conveniente poter agire a differenti livelli di astrazione. Dire che C# è un C++ "migliorato" non è corretto, è più corretto dire che C# è stato pensato per scopi differenti, usare l'uno o l'altro dipende da ciò che dobbiamo costruire.

venerdì 8 aprile 2011

Videosorveglianza fai-da-te!

Vediamo oggi come sia possibile creare una semplice applicazione per fare videosorveglianza “fai-da-te” completamente gratuita! Ciò che otterremo alla fine sarà una semplice applicazione che manda immagini  prese da una webcam ad un web server a cui possiamo collegarci per visualizzare cosa sta succedendo magari a casa nostra, in nostra assenza. Và detto che questa semplice applicazione è a solo scopo “didattico” e può essere arricchita a vostro piacimento.


Il nostro progetto è così architettato:

In pratica un “Frame Grabber” prende le immagini da un webcam e le salva su di un file, un “Image Sender” manda tali immagini ad un Web Server remoto che le pubblicherà. A questo punto le immagini della webcam saranno disponibili ad un qualsiasi browser remoto che si collega al web server permettendoci di fare videosorveglianza. Per divertirmi ho scelto di utilizzare linguaggi diversi per ciascun modulo:
  •  Frame Grabber: grabbing di immagini da webcam tramite librerie Open CV (C++)
  • Image Sender: classe Java che effettua una POST dell’immagine tramite una Http connection
  • Image Receiver: script PHP che salva sul web server l’immagine ricevuta
  • Web Page: pagina HTML che visualizza l’immagine

Come si nota, l’applicazione viene sviluppata parte in C/C++ , parte in PHP e parte in Java: in pratica una webcam riprende una scena, le immagini vengono grabbate e salvate ad intervalli regolari su un file del pc di casa tramite le librerie Open CV (librerie largamente utilizzate nel campo della computer vision). Una semplice applicazione Java, prende tale immagine e la spedisce (POST) ad un web server. Il web server a questo punto pubblicherà l’immagine tramite un semplice script PHP e sarà visibile tramite per esempio un semplice browser.
Per sviluppare l’applicazione ho usato ambienti di sviluppo per C/C++ e Java quali Visual Studio 2010 ed Eclipse, ma ciò non è vincolante poichè tutto il codice è completamente portabile su qualsiasi SO. Si noti che sul pc di casa (quello su cui gira l’Image Sender), è necessario avere installato una Java Virtual Machine scaricabile da qui.

Image Receiver (PHP)
Prima di tutto bisogna scegliere un web server (free!) che supporti Scripting PHP. Qui potete trovare una lista di esempio. Sceglietene uno, registratevi e createvi un sito. A questo punto mettete sul server (p.es. via FTP) il seguente file PHP:

recv.php
<?php
/**
 * Save the submitted file to disk
 */
if (is_uploaded_file($_FILES['img']['tmp_name'])) {
    move_uploaded_file($_FILES['img']['tmp_name'], "img.jpg");
}

Questo script non fa altro che salvare su disco l’immagine che proviene da un client tramite HTTP POST su un file “img.jpg”. In questo modo potete pubblicare un’immagine su web. Sostituite la pagina di default del sito con la seguente:

index.html
<html>
<head>
<title>Remote Surveillance System</title>
</head>
<body>

<img name="frame" src="" border="0" />

<script type="text/javascript">
    function showFrame() {
        document.images.frame.src = "img.jpg?" + Math.random();
        setTimeout("showFrame()", 3000);
    }
    showFrame();
</script>

</body>
</html>

In pratica la pagina effettua il refresh automatico dell’immagine “img.jpg” ogni 3 secondi escludendo la cache. Il file index.html dovrebbe essere la pagina di default del nostro sito e deve risiedere sulla stessa cartella del file recv.php. La parte da caricare sul web server è terminata. Passiamo ora ad implementare le 2 applicazioni che gireranno sul pc di casa.

Image Grabber (C/C++)
A questo punto siamo pronti per scrivere il codice che grabba le immagini da una semplice webcam e le salva su disco su un file “img.jpg”. Utilizzeremo le Open CV, librerie che vengono utilizzate per la computer vision. In pratica è un pò riduttivo usare le Open CV solo per grabbare delle immagini da una web cam, ma ciò potrebbe rivelarsi utile nel caso si voglia arricchire l’applicazione con funzionalità quali motion detection, object tracking ecc. Scarichiamo e compiliamo le Open CV dal sito. Per la compilazione delle librerie lib e dll possiamo utilizzare CMake o direttamente Visual Studio secondo queste istruzioni. Una volta che ci siamo installati le Open CV, creaiamo un nuovo progetto con Visual Studio, una semplice Console Application, chiamiamolo “VideoSorveglianza”. Includiamo le directory di include opportune di Open CV nella sezione “C/C++ -> Additional Include Directories” (p.es. “..\OpenCV2.2\include\opencv”) e tutte le librerie statiche nella sezione “Linker -> Input->Addicional Dependencies” (p.es. ..\lib\cv220.lib ecc.). Copiamo le librerie dinamiche delle Open CV (p.es. cv220.dll ecc.) nella cartella Debug e/o Release del nostro progetto.  Creiamo un nuovo file all’interno del progetto:

VideoSorveglianza.cpp

#include <stdio.h>
#include "cv.h" /* file include Open CV */
#include "highgui.h" /* file include Open CV */

int main(int argc, char** argv)
{
        CvCapture *capture;
        IplImage  *frame;
        double    t, ms = 0;
        int       key;

        /* inizializza la webcam */
        capture = cvCaptureFromCAM(0); /* cattura il flusso video dal primo device del pc*/
        cvNamedWindow("video", 1);

        while (key != 'q') {
                t = (double)cvGetTickCount();

                /* display a video */
                frame = cvQueryFrame(capture); /* grabba l’immagine */
                cvShowImage("video", frame);
                key = cvWaitKey(1); /* aspetta input da tastiera 1 sec */

                /* calcola il tempo trascorso */
                t = (double)cvGetTickCount() - t;
                ms += t/((double)cvGetTickFrequency() * 1000.0);

                /* autosave dell’immagine ogni 3 secondi */
                if (ceil(ms) >= 3000) {
                        cvSaveImage("img.jpg", frame);
                        ms = 0;
                }
        }

        /* free memory */
        cvReleaseCapture(&capture);
        cvDestroyWindow("video");

        return 0;
}

A questo punto, dopo aver compilato il progetto e collegato la webcam, possiamo lanciare l’eseguibile da console tramite il comando: “VideoSorveglianza.exe”. Il risultato è il seguente:


Tale immagine verrà salvata sul file locale “img.jpg” ad intervalli regolari di 3 secondi.

Image Sender (Java)
Il prossimo passo consiste nel creare il sender che spedirà le immagini ad un web server. Si noti che la spedizione deve avvenire come se fosse un HTTP POST. La classe Java (che girerà anch’essa sul nostro pc di casa), semplicemente legge l’immagine dal file “img.jpg”, apre una connessione URL col web server e la spedisce in formato binario. Apriamo Eclipse e creiamo un nuovo progetto, chiamiamolo p.es. “VideoSorveglianzaSender”. Creiamo una classe Sender con questo codice:

Sender.java

import java.net.*;
import java.awt.Graphics;
import java.awt.Image;
import java.awt.image.BufferedImage;
import java.io.*;

import javax.imageio.ImageIO;

public class Sender
{
    static URL u;
    private final String CrLf = "\r\n";
   
   
    void updateImage(){
       URLConnection conn = null;
       OutputStream os = null;
       InputStream is = null;
       try{
//Qui va sostituito l’indirizzo del web server: attenzione recv.php va lasciato
             URL url = new URL("http://<webserverURL>/recv.php");
             System.out.println("url:" + url);
             conn = url.openConnection();
             conn.setDoOutput(true);

// legge l’immagine da file e la mette su un array di byte
             File file = new File("img.jpg");
             Image image = ImageIO.read(file);
             BufferedImage bu = new
BufferedImage(image.getWidth(null),image.getHeight(null),BufferedImage.TYPE_INT_RGB);
             Graphics g = bu.getGraphics();
             g.drawImage(image,0,0,null);
             g.dispose();
             ByteArrayOutputStream baos = new ByteArrayOutputStream();
             ImageIO.write(bu, "jpg", baos);
             byte []imgData = baos.toByteArray();

// crea la request http

             String message1 = "";
             message1 += "-----------------------------4664151417711" + CrLf;
             message1 += "Content-Disposition: form-data; name=\"img\"; filename=\"img.jpg\"" + CrLf;
             message1 += "Content-Type: image/jpeg" + CrLf;
             message1 += CrLf;
      
             // l’immagine viene spedita come multipart message.
      
             String message2 = "";
             message2 += CrLf + "-----------------------------4664151417711--" + CrLf;
      
             conn.setRequestProperty("Content-Type", "multipart/form-data; boundary=---------------------------4664151417711");
             conn.setRequestProperty("Content-Length", String.valueOf((message1.length() + message2.length() + imgData.length)));
                   
             System.out.println("open os");
             os = conn.getOutputStream();
             System.out.println(message1);
             os.write(message1.getBytes());
      
             // SEND THE IMAGE
             int index = 0;
             int size = 1024;
             do{
                    System.out.println("write:" + index);
                    if((index+size)>imgData.length){
                           size = imgData.length - index;
                    }
                    os.write(imgData, index, size);
                    index+=size;
             }while(index<imgData.length);
                   
             System.out.println("written:" + index);
             System.out.println(message2);
             os.write(message2.getBytes());
             os.flush();
      
              // legge la risposta del web server
              System.out.println("open is");
             is = conn.getInputStream();
      
             char buff = 512;
             int len;
             byte []data = new byte[buff];
             do{
                    System.out.println("READ");
                    len = is.read(data);
                    if(len > 0){
                           System.out.println(new String(data, 0, len));
                    }
             }while(len>0);
      
             System.out.println("DONE");
      
       }catch(Exception e){
             e.printStackTrace();
       }finally{
             System.out.println("Close connection");
             try{
                    os.close();
             }catch(Exception e){}
             try{
                    is.close();
              }catch(Exception e){}
       }
    }
   
    public static void main(String args[]) throws IOException
    {
       while (true)
       {
             Sender send = new Sender();
             send.updateImage();       
             try {
// manda un immagine ogni 2 secondi
                    Thread.sleep(2000);
             } catch (InterruptedException e) {
                    e.printStackTrace();
             }
       }
    }
}

Sostituire nel punto segnato <webserverURL>, l’indirizzo URL del web server che abbiamo scelto all’inizio, avendo cura di chiamare lo script recv.php. La classe non fa altro che leggere il file img.jpg e metterlo dentro un byte array. A questo punto viene aperta una HTTP connection verso il web server e viene spedita l'immagine come "multipart message". L'image sender scriverà a video la risposta ottenuta dal web server.
A questo punto compiliamo la classe e copiamo il file “Sender.class” nella stessa directory Debug/Release di dove girerà il grabber “VideoSorveglianza.exe”.

Test finale
Proviamo il tutto: lanciamo l’eseguibile del grabber (dentro la cartella Debug/Release della solution di Visual Studio): “VideoSorveglianza.exe”. A questo punto lanciamo il sender tramite il comando “java Sender” lanciato dalla stessa cartella del progetto VideoSorveglianza. Aprendo un browser all’indirizzo del nostro sito vedremo le immagini provenienti dalla webcam che si aggiorneranno automaticamente. In questo modo potremmo controllare casa nostra magari direttamente dall’ufficio aprendo semplicemente un browser.

 
Design by Free WordPress Themes | Bloggerized by Lasantha - Premium Blogger Themes | cna certification