onsdag 28 november 2012
Web API Dependency Resolver
Denna artikel förklarar på ett bra sätt hur man använder Dependency Resolver i en MVC Web API Controller-klass.
Web API för nybörjare
Denna artikel ger en bra start att förstå Web API som introducerades i .NET MVC 4, och sedan även gå vidare att läsa denna artikel.
måndag 26 november 2012
Enkelt jQuery Ajax anrop i MVC
Att med hjälp av jQuery/Ajax anropa en C#-metod i ett MVC-projekt som returnerar Json är enkelt. I följande exempel ligger metoden i en controllerklass som heter HomeController, vilket ger URL:en /home/helloworld/ att anropa.
På sidan som ska anropa metoden har jag en DIV med id="responseResult" och lägger följande jQuery-kod.
Som bonus la jag till en fade-effekt efter 2 sekunder på resultatet.
Om man bara vill returnera en vanlig sträng kan koden i HelloWorld() se ut:
Då skrivs resultatet ut med endast "data" såhär:
public ActionResult HelloWorld()
{
return Json(new { Firstpart = "Hello", Secondpart = "World" }, JsonRequestBehavior.AllowGet);
}
På sidan som ska anropa metoden har jag en DIV med id="responseResult" och lägger följande jQuery-kod.
$(document).ready(function () {
$.ajax({
url: '/home/helloworld',
success: function (data) {
$('#responseResult').html('Resultat: ' + data.Firstpart + ' ' + data.Lastpart).delay(2000).fadeIn('slow');
},
error: function (data) {
alert('Error: Ajax call failed.');
}
});
});
Som bonus la jag till en fade-effekt efter 2 sekunder på resultatet.
Om man bara vill returnera en vanlig sträng kan koden i HelloWorld() se ut:
public ActionResult HelloWorld()
{
return Content("Hello World");
}
Då skrivs resultatet ut med endast "data" såhär:
$(document).ready(function () {
$.ajax({
url: '/home/helloworld',
success: function (data) {
$('#responseResult').html('Resultat: ' + data).delay(2000).fadeIn('slow');
},
error: function (data) {
alert('Error: Ajax call failed.');
}
});
});
fredag 16 november 2012
SimpleMembershipProvider WebSecurity.InitializeDatabaseConnection
Kom i kontakt med autentisering med hjälp av WebMatrix.WebData.SimpleMembershipProvider för att sköta inloggning och administration av användarkonton, men man gör det helst med hjälp av hjälpklassen WebSecurity.
Skapar man ett nytt default MVC 4 projekt i Visual Studio 2012 så skapas allt detta upp åt dig men med en lokal databasfil.
Om man som jag direkt vill ändra detta till att fungera mot sin SQL Server databas är det enkelt. Dock stötte jag på ett litet problem direkt om hur connectionsträngen skulle se ut för att metoden WebSecurity.InitializeDatabaseConnection() ska fungera.
Jag hade glömt att sätta attributet "providerName" till "System.Data.SqlClient":
<add name="MinDatabasConnection" providerName="System.Data.SqlClient" connectionString="Data Source=minDBserver\sqlexpress2012;Initial Catalog=DatabasNamnet;User ID=minDBuser;Password=*****" />
Sedan om inte de korrekta tabellerna redan existerar (vid ett nystartat projekt) så kan de begäras att skapas automatiskt via "autoCreateTable: true":
WebSecurity.InitializeDatabaseConnection("MinDatabasConnection", "UserProfile", "UserId", "UserName", autoCreateTables: true);
Skapar man ett nytt default MVC 4 projekt i Visual Studio 2012 så skapas allt detta upp åt dig men med en lokal databasfil.
Om man som jag direkt vill ändra detta till att fungera mot sin SQL Server databas är det enkelt. Dock stötte jag på ett litet problem direkt om hur connectionsträngen skulle se ut för att metoden WebSecurity.InitializeDatabaseConnection() ska fungera.
Jag hade glömt att sätta attributet "providerName" till "System.Data.SqlClient":
<add name="MinDatabasConnection" providerName="System.Data.SqlClient" connectionString="Data Source=minDBserver\sqlexpress2012;Initial Catalog=DatabasNamnet;User ID=minDBuser;Password=*****" />
Sedan om inte de korrekta tabellerna redan existerar (vid ett nystartat projekt) så kan de begäras att skapas automatiskt via "autoCreateTable: true":
WebSecurity.InitializeDatabaseConnection("MinDatabasConnection", "UserProfile", "UserId", "UserName", autoCreateTables: true);
söndag 11 november 2012
XML serialisering av C# objekt
Detta exempel är en bra början att lära sig hur XML serialisering fungerar.
Om man t.ex. vill spara ner ett eller flera objekt till en XML-fil för att senare läsa tillbaka den igen.
Om man t.ex. vill spara ner ett eller flera objekt till en XML-fil för att senare läsa tillbaka den igen.
fredag 9 november 2012
SignalR chat example
Nyheter i VS2012 och .NET 4.5
Bra artikel av Scott Hanselman med videos om nyheterna i Visual Studio 2012 och .NET 4.5
Läs/se den här.
Läs/se den här.
torsdag 8 november 2012
.Net using Javascript/ASMX-Webservice
Snabb demo av hur javascript och WebService(asmx)-anrop funkar i .NET
Fil Demo.asmx
På samma ASPX-sida som du kommer ha ditt javascript, behöver du lägga till en ScriptManager:
För denna demo lägger jag till en SPAN-tagg där resultatet från anropet kommer visas:
Själva javascriptanropet kan nu göras på följande sätt:
Fil Demo.asmx
namespace TestProject.AsmxServices
{
/// <summary>
/// Summary description for Demo
/// </summary>
[WebService(Namespace = "http://tempuri.org/")]
[WebServiceBinding(ConformsTo = WsiProfiles.BasicProfile1_1)]
[System.ComponentModel.ToolboxItem(false)]
[System.Web.Script.Services.ScriptService]
public class Demo : System.Web.Services.WebService
{
[WebMethod]
public string HelloWorld()
{
return "Hello World";
}
}
}
På samma ASPX-sida som du kommer ha ditt javascript, behöver du lägga till en ScriptManager:
<asp:ScriptManager runat="server">
<Services>
<asp:ServiceReference Path="/AsmxServices/Demo.asmx" />
</Services>
</asp:ScriptManager>
För denna demo lägger jag till en SPAN-tagg där resultatet från anropet kommer visas:
<span id="asmxOutPut" style="font-weight: bold;"></span>
Själva javascriptanropet kan nu göras på följande sätt:
TestProject.AsmxServices.Demo.HelloWorld(_onCallbackSuccess, _onRequestFailed);
function _onCallbackSuccess(result, userContext) {
document.getElementById('asmxOutPut').innerHTML = result;
}
function _onRequestFailed(result, userContext) {
alert('Request failed');
}
tisdag 6 november 2012
Cache av resultat från klassmetoder
I många klassmetoder så gör man tunga databasfrågor eller andra prestandakrävande operationer som ofta resulterar i samma resultat eftersom inget har ändrats i datat sedan sist denna metod kördes.
Då kan det vara en bra idé att resultatet läggs i cachminnet under en tidsperiod som man tycker är rimligt.
Låt säga att att en metod efterfrågas varje gång första sidan av din webbplats anropas, och denna metod kräver en eller flera databasanrop för att kunna presentera resultatet på sidan. Men eftersom datat för sidan uppdateras i regel som oftast kanske bara en gång var 5:e minut, eller ännu mer sällan så räcker det med att hämta datat en gång från databasen och sedan låta det datat vara ok att presentera (från cacheminnet) på första sidan i, låt säga en minut, oavsett om datat har uppdaterats i databasen.
I följande fåniga exempel visar jag hur man kan cache'a resultatet av en klass olika metoder med olika unika strängnycklar som identifierar resultatet beroende på klass-, metodnamn och metodens olika inparametrar.
Först har jag en hjälpklass med metoden GetCacheKey() som kan användas för alla olika klasser:
På sidan har jag sedan placerat en Literal kallas litOut.
Jag har också skapat en liten Car-klass som har en metod GetCarStamp() som ger tillbaka en specifik sträng för objektet.
Vi ponerar att denna metod kräver prestandakrävande databasanrop så därför är värdet lagrat i cacheminnet i 1minut innan man återigen måste läsa upp det från databasen.
Page_Load metoden ser ut som följer:
Man skulle då få tillbaka strängen "ABC123BLUE" i en minut tills cachen släpper, även om färgen eller registreringsnumret eventuellt har ändrats i databasen under denna tid.
På detta vis kan man göra likadant för alla andra prestandakrävande metoder som klassen eventuellt innehåller och spara dess resultat i cachen med klassnamn (ClassCacheKey) och metodnamn som unik identifierare.
I min Helper-fil har jag lagt till följande statiska metoder:
Och min Car-klass ser nu ut såhär:
Så cachen för GetCarStamp() släpper inte förrän färgen ändras via UpdateCarColor(), eller att applikationen startas om.
Vill man att den ändå ska släppa efter en viss tid så är det bara att byta ut Cache.NoAbsoluteExpiration till det värde man önskar.
Då kan det vara en bra idé att resultatet läggs i cachminnet under en tidsperiod som man tycker är rimligt.
Låt säga att att en metod efterfrågas varje gång första sidan av din webbplats anropas, och denna metod kräver en eller flera databasanrop för att kunna presentera resultatet på sidan. Men eftersom datat för sidan uppdateras i regel som oftast kanske bara en gång var 5:e minut, eller ännu mer sällan så räcker det med att hämta datat en gång från databasen och sedan låta det datat vara ok att presentera (från cacheminnet) på första sidan i, låt säga en minut, oavsett om datat har uppdaterats i databasen.
I följande fåniga exempel visar jag hur man kan cache'a resultatet av en klass olika metoder med olika unika strängnycklar som identifierar resultatet beroende på klass-, metodnamn och metodens olika inparametrar.
Först har jag en hjälpklass med metoden GetCacheKey() som kan användas för alla olika klasser:
namespace CacheTestProject.Helpers
{
public class Helper
{
public static string GetCacheKey(string className, string methodName, params string[] parameters)
{
return className + "-" + methodName + "-" + string.Join("-", parameters);
}
}
}
På sidan har jag sedan placerat en Literal kallas litOut.
Jag har också skapat en liten Car-klass som har en metod GetCarStamp() som ger tillbaka en specifik sträng för objektet.
Vi ponerar att denna metod kräver prestandakrävande databasanrop så därför är värdet lagrat i cacheminnet i 1minut innan man återigen måste läsa upp det från databasen.
public class Car
{
private const string ClassCacheKey = "Car";
public string RegNumber { get; set; }
public string Color { get; set; }
public Car(string regnr, string color)
{
RegNumber = regnr;
Color = color;
}
// EXAMPLE METHOD 1 without cachedependency!
public string GetCarStamp(bool withRegNumber, bool withColor)
{
string cacheKey = Helper.GetCacheKey(
ClassCacheKey,
"GetCarStamp",
RegNumber);
string data = (string)HttpRuntime.Cache.Get(cacheKey);
if (!string.IsNullOrEmpty(data))
{
return data;
}
//Call for the performance demanding method as nothing is in the cache!
data = GetCarStampFromDatabase();
HttpRuntime.Cache.Insert(
cacheKey,
data,
null,
DateTime.Now.AddSeconds(60),
Cache.NoSlidingExpiration);
return data;
}
//Simulate a performance demanding method that we want to
//avoid calling as much as possible
private string GetCarStampFromDatabase()
{
var regNumber = (withRegNumber) ? RegNumber : string.Empty;
var color = (withColor) ? Color : string.Empty;
return string.Format("{0}{1}", RegNumber, color);
}
}
Page_Load metoden ser ut som följer:
protected void Page_Load(object sender, EventArgs e)
{
Car MyCar1 = new Car("ABC123", "BLUE");
litOut.Text = MyCar1.GetCarStamp();
}
Man skulle då få tillbaka strängen "ABC123BLUE" i en minut tills cachen släpper, även om färgen eller registreringsnumret eventuellt har ändrats i databasen under denna tid.
På detta vis kan man göra likadant för alla andra prestandakrävande metoder som klassen eventuellt innehåller och spara dess resultat i cachen med klassnamn (ClassCacheKey) och metodnamn som unik identifierare.
Med Dependency
Om man istället vill att resultatet ska ligga kvar i cachen så länge som inte någonting ändras, alltså att resultatet ska invalideras vid någon form av uppdatering på objektet, kan man justera koden till något i stil med nedanstående.I min Helper-fil har jag lagt till följande statiska metoder:
public static string GetCacheDependencyKey(string className, string SomeUniqueId)
{
return className + "-CacheDepKey-" + SomeUniqueId;
}
public static CacheDependency GetCacheDependency(string className, string SomeUniqueId)
{
//Get unique identification key for the dependency
string DependencyKey = GetCacheDependencyKey(className, SomeUniqueId);
//If it is not already in the cache, add it!
if (HttpRuntime.Cache.Get(DependencyKey) == null)
{
HttpRuntime.Cache.Insert( //HttpRuntime.Cache.Insert will replace object in cache, which HttpRuntime.Cache.Add does not!
DependencyKey,
string.Empty,
null,
Cache.NoAbsoluteExpiration,
Cache.NoSlidingExpiration);
}
//Then create a dependency that tells to depend on the cacheitem with key "DependencyKey"
return new CacheDependency(
null,
new[] { DependencyKey }
);
}
Och min Car-klass ser nu ut såhär:
public class Car
{
private const string ClassCacheKey = "Car";
public string RegNumber { get; set; }
public string Color { get; set; }
public Car(string regnr, string color)
{
RegNumber = regnr;
Color = color;
}
public void UpdateCarColor(string newColor)
{
Color = newColor;
HttpRuntime.Cache.Remove(
Helper.GetCacheDependencyKey(
ClassCacheKey, RegNumber
)
);
}
// EXAMPLE METHOD 1 without cachedependency!
public string GetCarStamp()
{
string cacheKey = Helper.GetCacheKey(
ClassCacheKey,
"GetCarStamp",
RegNumber);
string data = (string)HttpRuntime.Cache.Get(cacheKey);
if (!string.IsNullOrEmpty(data))
{
return data;
}
//Call for the performance demanding method...
data = GetCarStampFromDatabase();
HttpRuntime.Cache.Insert(
cacheKey,
data,
Helper.GetCacheDependency(ClassCacheKey, RegNumber),
Cache.NoAbsoluteExpiration,
Cache.NoSlidingExpiration);
return data;
}
//Simulate a performance demanding method that we want to
//avoid calling as much as possible
private string GetCarStampFromDatabase()
{
return string.Format("{0}{1}", RegNumber, Color);
}
}
Så cachen för GetCarStamp() släpper inte förrän färgen ändras via UpdateCarColor(), eller att applikationen startas om.
Vill man att den ändå ska släppa efter en viss tid så är det bara att byta ut Cache.NoAbsoluteExpiration till det värde man önskar.
Sveriges bästa systemutvecklare 2012
Inte mycket för att tävla i kompetens och prestation inom yrkeslivet, men denna lista från IDG.se ger ändå en intressant bild av vad andra utvecklare arbetar med.
Jag har själv haft förmånen att få arbeta med Joel Abrahamsson (plats 8, plats 5 år 2011) som är en mycket trevlig och kompetent utvecklare inom .NET och EPiServer.
Se hela listan här.
Jag har själv haft förmånen att få arbeta med Joel Abrahamsson (plats 8, plats 5 år 2011) som är en mycket trevlig och kompetent utvecklare inom .NET och EPiServer.
Se hela listan här.
måndag 5 november 2012
Cache in .NET MVC
Att sätta cache på en sida eller en del av en sida i MVC görs på samma sätt som i WebForms - via attribut i koden på controllerklassen:
Om man vill ange flera parameternamn i VaryByParam ska de åtskiljas av semikolon.
public class HomeController : Controller
{
//Cache page for "Duration" amount of seconds
[OutputCache(Duration = 5, VaryByParam = "none")]
public ActionResult Index()
{
// Write your code here
return View();
}
}
Om man vill ange flera parameternamn i VaryByParam ska de åtskiljas av semikolon.
DropDownList in MVC
Ett litet exempel på hur man implementerar en DropDownList i .NET MVC med Razor syntax så det ser ut enligt nedan. Jag visar också hur man loopar igenom objekten i en foreach-loop:
I controller-metoden har jag en lista av klasstypen "Room".
Filen för vyn ser ut enligt följande:
I controller-metoden har jag en lista av klasstypen "Room".
public ActionResult Index(){
List<Room> myRoomList = new List<Room>() {
new Room() { Roomname = "Horror room", Height = 5 },
new Room() { Roomname = "Death room", Height = 6 }
};
SelectList myList = new SelectList(myRoomList, "Height", "Roomname");
ViewBag.DropdownRoomList = myList;
return View(myRoomList);
}
Filen för vyn ser ut enligt följande:
@model IList<DropDownListTestProject.Models.Room>
@{
ViewBag.Title = "DropDownList test";
}
@foreach (var amodel in Model)
{
@: Height of @amodel.Roomname is @amodel.Height metres
<br />
}
<br />
@Html.DropDownList("RoomHeight", (SelectList)ViewBag.DropdownRoomList)
lördag 3 november 2012
Chicken Tikka Masala
4 personer
- 500 gram kycklingfiléer (bröst el lår)
- 1 st gul lök
- 1 st vitlöksklyfta
- 1 msk färsk ingefära, finhackad
- 1 st påse tikka masala-krydda (40 g)
- 1,5 dl sötmandel
- 400 gram krossade tomater
- 1 dl vispgrädde
- 1 dl matyoghurt (10 %)
- Skär kycklingfiléerna (om det är bröst) i bitar, ca 4 cm. Strimla lök och vitlök. Bryn i smör- & rapsolja i stekpanna eller kastrull. Salta och peppra.
- Blanda i ingefära, tikka masala-krydda, sötmandel och krossade tomater. Låt koka under lock ca 10 minuter.
- Ta ur kycklingbitarna och lägg på ett fat. Mixa såsen med stavmixer eller i matberedare tills den blir nästan slät. Lägg i kycklingen igen och tillsätt grädde och yoghurt. Låt koka ytterliggare 2 minuter, men akta så det inte bränner i botten!
- Servera med basmatiris, en sallad och nygräddade naan-bröd!
söndag 28 oktober 2012
Singleton in C#
En singleton i C# kan se ut något i stil med följande:
Eller med hjälp av klassen Lazy<T>():
Läs mer om singleton på denna sida:
http://www.yoda.arachsys.com/csharp/singleton.html
public sealed class Singleton
{
private static Singleton instance = null;
private static readonly object padlock = new object();
private Singleton()
{
}
public static Singleton Instance
{
get
{
lock (padlock)
{
if (instance==null)
{
instance = new Singleton();
}
return instance;
}
}
}
}
Eller med hjälp av klassen Lazy<T>():
public class Singleton
{
private readonly static Lazy _instance = new Lazy(() => new Singleton());
public static Singleton Instance
{
get
{
return _instance.Value;
}
}
}
Läs mer om singleton på denna sida:
http://www.yoda.arachsys.com/csharp/singleton.html
EnableViewState - true or false?
När är det nödvändigt att aktivera eller avaktivera EnableViewState på sidor och kontroller?
Bra info på denna länk om detta.
Bra info på denna länk om detta.
AutoEventWireup - true or false?
Klockren info på nedanstående länk om ordningsföljden på hur sidors/kontrollers metoder parsas.
http://hockeswe.blogspot.se/2007/07/autoeventwireuptrue.html
TemplateControl.HookUpAutomaticHandlers (vilken anropas under kontrollens OnInit bland annat) kommer om AutoEventWireup=true att försöka hitta följande metoder:
Om kontrollen är Page:
Page_PreInit
Page_PreLoad
Page_LoadComplete
Page_PreRenderComplete
Page_InitComplete
Page_SaveStateComplete
För alla TemplateControls:
Page_Init
Page_Load
Page_DataBind
Page_PreRender
Page_Unload
Page_Error
Page_AbortTransaction eller om den inte finns; OnTransactionAbort
Page_CommitTransaction eller om den inte finns; OnTransactionCommit
Detta görs i metoden TemplateControl.GetDelegateInformationWithNoAssert().
http://hockeswe.blogspot.se/2007/07/autoeventwireuptrue.html
TemplateControl.HookUpAutomaticHandlers (vilken anropas under kontrollens OnInit bland annat) kommer om AutoEventWireup=true att försöka hitta följande metoder:
Om kontrollen är Page:
Page_PreInit
Page_PreLoad
Page_LoadComplete
Page_PreRenderComplete
Page_InitComplete
Page_SaveStateComplete
För alla TemplateControls:
Page_Init
Page_Load
Page_DataBind
Page_PreRender
Page_Unload
Page_Error
Page_AbortTransaction eller om den inte finns; OnTransactionAbort
Page_CommitTransaction eller om den inte finns; OnTransactionCommit
Detta görs i metoden TemplateControl.GetDelegateInformationWithNoAssert().
måndag 23 juli 2012
Proffsiga videotutorials
Det finns en uppsjö av olika videotutorials inom utveckling men jag har nu upptäckt http://www.pluralsight-training.net som producerar väldigt bra och proffsiga videotutorials.
De tar betalt men inom mitt intresseområde fanns några gratis att nå från http://www.asp.net/mvc som beskrev på ett bra sätt olika delar av MVC-utveckling inom .Net, Ajax, C# m.m.
Dessutom kan man jämföra det med att köpa dyr litteratur som man måste läsa igenom, eller att bli sittandes och försöka Googla fram information när man behöver den.
Alltså bara att luta sig tillbaka och njuta av föreläsningar inom ett specifikt område man vill lära sig.
De tar betalt men inom mitt intresseområde fanns några gratis att nå från http://www.asp.net/mvc som beskrev på ett bra sätt olika delar av MVC-utveckling inom .Net, Ajax, C# m.m.
Dessutom kan man jämföra det med att köpa dyr litteratur som man måste läsa igenom, eller att bli sittandes och försöka Googla fram information när man behöver den.
Alltså bara att luta sig tillbaka och njuta av föreläsningar inom ett specifikt område man vill lära sig.
torsdag 23 februari 2012
Steget mot busskörkort nu taget
Idag skrev jag in mig på Grönlunds trafikskola i Haninge för att skaffa mig D-behörighet - alltså busskörkort.
Jag har funderat på detta ett bra tag och nu har jag tiden och chansen att få det gjort, så varför inte. Kom ju tidigare i höstas in på skolan för bussförarutbildningen men tackade nej då jag redan börjat min utbildning till behandlingspedagog. Dessutom fick jag även jobb via Nobina som jag sökt men tackade även där nej av samma anledning. I båda fallen hade jag ju fått busskörkortet gratis - förutom studielån osv.
Men detta alternativ blir en bra lösning för min del och hur jag tänker om framtiden.
Jag har funderat på detta ett bra tag och nu har jag tiden och chansen att få det gjort, så varför inte. Kom ju tidigare i höstas in på skolan för bussförarutbildningen men tackade nej då jag redan börjat min utbildning till behandlingspedagog. Dessutom fick jag även jobb via Nobina som jag sökt men tackade även där nej av samma anledning. I båda fallen hade jag ju fått busskörkortet gratis - förutom studielån osv.
Men detta alternativ blir en bra lösning för min del och hur jag tänker om framtiden.
Började med ett startpaket där teorilektioner, böcker, 10 internettester och 5 körlektioner ingår för 12500:-.
Sedan kommer en massa annat att tillkomma så som ytterligare körlektioner, uppkörningskostnader och teoriprov etc. Dessutom den s.k. YKB (Yrkeskompetensbevis) så man får köra yrkestrafik.
Tanken är att jag kommer antingen ha nytta av körkortet i jobbet på behandlingshem, men främst att jag kan ta extrajobb senare och köra turisttrafik.
Så nu vet jag vad jag ägnar min lediga tid åt i vår och sommar...
Prenumerera på:
Inlägg (Atom)