C#中的類型轉(zhuǎn)換-自定義隱式轉(zhuǎn)換和顯式轉(zhuǎn)換

來源:https://note.guoqianfan.com/2022/04/22/operator-implicit-explicit-in-csharp/

前言

有時(shí)我們會(huì)遇到這么一種情況:在json數(shù)據(jù)里翎蹈,數(shù)組里的數(shù)據(jù)類型不一致慢睡,導(dǎo)致我們不能直接反序列化為目標(biāo)類型传睹。最終我們只能反序列化為JObject類型据悔,然后通過字符串取值的方式來取出數(shù)據(jù)。

下面介紹一種新方式:通過自定義隱式轉(zhuǎn)換导犹,把不一樣的數(shù)據(jù)類型反序列化為一樣的數(shù)據(jù)類型尔艇。

基礎(chǔ)知識(shí)

類型轉(zhuǎn)換有2種:隱式轉(zhuǎn)換和顯式轉(zhuǎn)換。但是喜庞,不管是隱式轉(zhuǎn)換诀浪,還是顯式轉(zhuǎn)換棋返,都是生成了一個(gè)新對(duì)象返回的。改變新對(duì)象的屬性雷猪,不會(huì)影響老對(duì)象懊昨!(dynamic對(duì)象除外,詳情搜索dynamic動(dòng)態(tài)類型春宣。)

自定義隱式/顯式轉(zhuǎn)換的方法需要用到幾個(gè)關(guān)鍵字:implicit(隱式轉(zhuǎn)換)酵颁、explicit(顯式轉(zhuǎn)換)、operator(操作符)月帝。更多的注意點(diǎn)見下:

  1. 方法必須是static
  2. 使用implicitexplicit
  3. 搭配operator(此也是c#關(guān)鍵字躏惋,可在類別或結(jié)構(gòu)宣告內(nèi)多載內(nèi)建運(yùn)算子或提供使用者定義的轉(zhuǎn)換)
  4. 返回值為要轉(zhuǎn)換為的目標(biāo)類型,但不要在方法上聲明嚷辅,方法名目標(biāo)類型簿姨。注意:返回值不一定是本類類型。本類型和其他類型之間可以互相轉(zhuǎn)換簸搞,只要定義轉(zhuǎn)換方法就行扁位。
  5. 參數(shù)為原始類型,方法名目標(biāo)類型
  6. 類A到類B的類型轉(zhuǎn)換定義不能在類C中進(jìn)行(即2個(gè)類的轉(zhuǎn)換不能在第3個(gè)類中定義)趁俊,否則會(huì)報(bào)錯(cuò):用戶定義的轉(zhuǎn)換必須是轉(zhuǎn)換成封閉類型域仇,或者從封閉類型轉(zhuǎn)換。具體查看后面的用戶定義的轉(zhuǎn)換必須是轉(zhuǎn)換成封閉類型寺擂,或者從封閉類型轉(zhuǎn)換
  7. 不能被virtual/override修飾(不能“覆蓋”運(yùn)算符暇务,因?yàn)樗鼈兪庆o態(tài)的。)Overriding implicit operators in C#

示例代碼

//================定義類型和方法================
class Robot
{
    public int Id { get; set; }
    public string Name { get; set; }

    public Robot(int id, string name)
    {
        Id = id;
        Name = name;
    }

    #region 其他類型->本類

    //隱式轉(zhuǎn)換
    public static implicit operator Robot(string name)
    {
        return new Robot(101, name);
    }

    //顯式轉(zhuǎn)換
    public static explicit operator Robot(int id)
    {
        return new Robot(id, "miku");
    }

    #endregion

    #region 本類->其他類型

    //隱式轉(zhuǎn)換
    public static implicit operator string(Robot robot)
    {
        return robot.Name;
    }

    //顯式轉(zhuǎn)換
    public static explicit operator int(Robot robot)
    {
        return robot.Id;
    }

    #endregion
}

//================測(cè)試代碼================
#region 其他類型->本類

string gumiStr = "gumi";
Robot gumi001 = gumiStr; //隱式轉(zhuǎn)換
Console.WriteLine("隱式轉(zhuǎn)換:gumi001 : {0}", JsonConvert.SerializeObject(gumi001));

int lukaId = 1004;
Robot luka001 = (Robot)lukaId; //顯式轉(zhuǎn)換
Console.WriteLine("顯式轉(zhuǎn)換:luka001 : {0}", JsonConvert.SerializeObject(luka001));

#endregion

#region 其他類型->本類

Robot miku001 = new Robot(1001, "miku10001");
//隱式轉(zhuǎn)換
string mikuName = miku001;
//顯式轉(zhuǎn)換
int mikuId = (int)miku001;

Console.WriteLine("隱式轉(zhuǎn)換:miku001 Name: {0}", mikuName);
Console.WriteLine("顯式轉(zhuǎn)換:miku001 Id: {0}", mikuId);

#endregion

輸出結(jié)果如下:

隱式轉(zhuǎn)換:gumi001 : {"Id":101,"Name":"gumi"}
顯式轉(zhuǎn)換:luka001 : {"Id":1004,"Name":"miku"}
隱式轉(zhuǎn)換:miku001 Name: miku10001
顯式轉(zhuǎn)換:miku001 Id: 1001

實(shí)際應(yīng)用

問題

[1,[[2,2],[2,2],[2,2],[2,2]]]

這樣一個(gè)字符串怔软,如何可以反序列化成一個(gè)對(duì)象垦细?(如何定義這個(gè)類?)

答案

using System;
using System.Linq;
using System.Collections.Generic;
using Newtonsoft.Json;
using Newtonsoft.Json.Linq;
                    
public class Program
{
    public static void Main()
    {
        var json = "[1,[[2,2],[2,2],[2,2],[2,2]]]";
        var root = JsonConvert.DeserializeObject<Root>(json);
        foreach(var ele in root)
        {
            if(ele.SingleValue.HasValue)
            {//有值挡逼,原始數(shù)據(jù)為 1
                Console.WriteLine(ele.SingleValue.Value);
            }else
            {//原始數(shù)據(jù)為 二維數(shù)組
                Console.WriteLine(string.Join(" ",ele.Select(x=>string.Join(",",x))));
            }
        }
        Console.WriteLine(JsonConvert.SerializeObject(root));
    }
}

class Root : List<Element> { }
[JsonConverter(typeof(CConverter))]
class Element : List<List<long>>
{
    //該屬性括改,存放 1 。后續(xù)可以通過判斷該屬性是否有值來得知原始數(shù)據(jù)的情況
    public long? SingleValue { get; set; }

    //遇到 1 家坎,隱式轉(zhuǎn)換為 該類型嘱能,其中 1 被存放到SingleValue屬性
    public static implicit operator Element(long d)
    {
        return new Element { SingleValue = d };
    }
}

public class CConverter : JsonConverter
{
    public override bool CanConvert(Type objectType)
    {
        return (objectType == typeof(Element));
    }

    public override bool CanRead  { get { return false; } }
    public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
    {
        throw new NotImplementedException();
    }

    public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
    {
        var ele = value as Element;
        var token = ele.SingleValue.HasValue ? JToken.FromObject(ele.SingleValue.Value) : JToken.FromObject(ele.ToList());
        token.WriteTo(writer);
    }

    public override bool CanWrite { get { return true; } }
}

報(bào)錯(cuò)

用戶定義的轉(zhuǎn)換必須是轉(zhuǎn)換成封閉類型,或者從封閉類型轉(zhuǎn)換

這個(gè)錯(cuò)誤乘盖,與封閉類型無關(guān)焰檩。

是因?yàn)橛羞@個(gè)限制:類A到類B的類型轉(zhuǎn)換定義不能在類C中進(jìn)行(即2個(gè)類的轉(zhuǎn)換不能在第3個(gè)類中定義)

所以對(duì)于目標(biāo)類型是集合類List<T>,我們無法直接定義到它的轉(zhuǎn)換订框。不過析苫,有2個(gè)迂回的方法:

  • 創(chuàng)建個(gè)類繼承自集合類List<T>,定義到這個(gè)子類的轉(zhuǎn)換。上面實(shí)際應(yīng)用中的代碼就是這樣做的:class Element : List<List<long>>
  • 創(chuàng)建T1T2的自定義轉(zhuǎn)換衩侥,使用時(shí)逐個(gè)轉(zhuǎn)換:list.Select(p=>(B)p).ToList()国旷。

參考

  1. 隱式轉(zhuǎn)換:用戶定義的轉(zhuǎn)換必須是轉(zhuǎn)換成封閉類型,或者從封閉類型轉(zhuǎn)換:https://blog.csdn.net/kamui_shiron/article/details/8807142

其他

應(yīng)用和設(shè)計(jì)

在定義類別時(shí)茫死,如果有需要跪但,就可以使用這兩個(gè)關(guān)鍵字來提供類別一些額外的功能

但在使用時(shí)也必須考慮設(shè)計(jì)上是否合理

例如當(dāng)兩類別有相關(guān)性時(shí)是否該提取出父類或是接口來使用,而不是為了方便做了一堆轉(zhuǎn)換峦萎,導(dǎo)致程式撰寫與維護(hù)上的困難屡久。

讀音

  • 隱式轉(zhuǎn)換:implicit [?m?pl?s?t] adj.不言明[含蓄]的; 無疑問的,絕對(duì)的; 成為一部份的; 內(nèi)含的;
  • 顯式轉(zhuǎn)換:explicit [?k?spl?s?t] adj.明確的爱榔,清楚的; 直言的; 詳述的; 不隱瞞的;

參考

  1. 【問】這樣一個(gè)字符串如何反序列化:http://www.newsmth.net/nForum/#!article/DotNET/69817
  2. 型別轉(zhuǎn)換關(guān)鍵字explicit與implicit的用法:https://dotblogs.com.tw/lastsecret/2011/11/14/57875
  3. c#關(guān)鍵詞implicit和explicit:https://blog.csdn.net/Joyhen/article/details/40110391
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末被环,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子详幽,更是在濱河造成了極大的恐慌筛欢,老刑警劉巖,帶你破解...
    沈念sama閱讀 221,198評(píng)論 6 514
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件唇聘,死亡現(xiàn)場(chǎng)離奇詭異版姑,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)迟郎,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,334評(píng)論 3 398
  • 文/潘曉璐 我一進(jìn)店門剥险,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人谎亩,你說我怎么就攤上這事炒嘲∮钜Γ” “怎么了匈庭?”我有些...
    開封第一講書人閱讀 167,643評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)浑劳。 經(jīng)常有香客問我阱持,道長(zhǎng),這世上最難降的妖魔是什么魔熏? 我笑而不...
    開封第一講書人閱讀 59,495評(píng)論 1 296
  • 正文 為了忘掉前任衷咽,我火速辦了婚禮,結(jié)果婚禮上蒜绽,老公的妹妹穿的比我還像新娘镶骗。我一直安慰自己,他們只是感情好躲雅,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,502評(píng)論 6 397
  • 文/花漫 我一把揭開白布鼎姊。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪相寇。 梳的紋絲不亂的頭發(fā)上慰于,一...
    開封第一講書人閱讀 52,156評(píng)論 1 308
  • 那天,我揣著相機(jī)與錄音唤衫,去河邊找鬼婆赠。 笑死,一個(gè)胖子當(dāng)著我的面吹牛佳励,可吹牛的內(nèi)容都是我干的休里。 我是一名探鬼主播,決...
    沈念sama閱讀 40,743評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼赃承,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼份帐!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起楣导,我...
    開封第一講書人閱讀 39,659評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤废境,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后筒繁,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體噩凹,經(jīng)...
    沈念sama閱讀 46,200評(píng)論 1 319
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,282評(píng)論 3 340
  • 正文 我和宋清朗相戀三年毡咏,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了驮宴。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 40,424評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡呕缭,死狀恐怖堵泽,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情恢总,我是刑警寧澤迎罗,帶...
    沈念sama閱讀 36,107評(píng)論 5 349
  • 正文 年R本政府宣布,位于F島的核電站片仿,受9級(jí)特大地震影響纹安,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜砂豌,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,789評(píng)論 3 333
  • 文/蒙蒙 一厢岂、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧阳距,春花似錦塔粒、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,264評(píng)論 0 23
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)映跟。三九已至,卻和暖如春扬虚,著一層夾襖步出監(jiān)牢的瞬間努隙,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,390評(píng)論 1 271
  • 我被黑心中介騙來泰國(guó)打工辜昵, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留荸镊,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,798評(píng)論 3 376
  • 正文 我出身青樓堪置,卻偏偏與公主長(zhǎng)得像躬存,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子舀锨,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,435評(píng)論 2 359

推薦閱讀更多精彩內(nèi)容