如何将obj列表转换为int类型。我试图使用下面的地图函数添加两个列表,但它不适用于obj列表。
let query f=
seq{
let cmd = new OleDbCommand( "SELECT * FROM F" );
let conn = new OleDbConnection( @"Provider=Microsoft.ACE.OLEDB.12.0;
Data Source=D:\Users\df\Documents\Vfolio.accdb;
Persist Security Info=False;" )
conn.Open()
let DAdapt = new OleDbDataAdapter("SELECT * FROM F",conn)
let DTab = new DataSet()
let i= DAdapt.Fill(DTab)
let rowCol = DTab.Tables.[0].Rows
let rowCount = rowCol.Count
for i in 0 .. (rowCount - 1) do
yield f (rowCol.[i])
}
let u= query(fun row -> row.[0])
let a= List.ofSeq u
let v=query(fun row -> row.[1])
let b= List.ofSeq v
let c = List.map2 (fun x y-> x + y) a b
错误消息:类型'obj'不支持运算符'+'
答案 0 :(得分:5)
由于row.[i]
会返回obj
类型,您的u
和v
会变为seq<obj>
,从而a
和b
变为List<obj>
类型,因此x
和y
被推断为类型为obj
,当然,您无法添加两个obj
s ,这正是编译器告诉你的。
如果您确定row.[0]
和row.[1]
是某种数字,则应该应用适当的演员,例如:
let u= query(fun row -> row.[0] :?> int)
let a= List.ofSeq u
let v=query(fun row -> row.[1] :?> int)
let b= List.ofSeq v
let c = List.map2 (fun x y-> x + y) a b
您也可以在其他地方应用此演员表,具体取决于您的品味和要求,例如:
let c = List.map2 (fun x y-> (x :?> int) + (y :?> int)) a b
或者:
let a= u |> Seq.cast<int> |> List.ofSeq
let b= v |> Seq.cast<int> |> List.ofSeq
但是我喜欢第一个最好的例子,因为它在最早的已知点应用了强制转换,并且产生了最少量的额外代码。
但要注意:如果row.[0]
在运行时变为int
,则会得到InvalidCastException
。
P.S。在List.map2
来电中,您可以直接指定(+)
,而不是将其包装在额外的lambda中:
List.map2 (+) a b
P.P.S 此外,您的List.ofSeq
来电似乎很浪费,因为Seq
也有map2
:
let u = query(fun row -> row.[0] :?> int)
let v = query(fun row -> row.[1] :?> int)
let c = Seq.map2 (+) u v |> List.ofSeq
P.P.P.S 另外,您是否注意到对query
的两次调用都会生成自己的数据库连接,命令,适配器和数据集?您是否打算这样做或者您的意思是只有一个连接然后从结果中获取不同的列?如果是这样,您应该只拨打query
一次:
let c = query( fun row -> (row.[0] :?> int) + (row.[1] :?> int) ) |> List.ofSeq